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内 容 提 要 


本 书 分 为 四 部 分 ， 第 一 部 分 全 面 介 绍 Java 7 的 新 特性 ， 第 二 部 分 探讨 Java 关键 编程 知识 和 技术 ， 第 三 
部 分 讨论 JVYM 上 的 新 语言 和 多 语言 编程 ， 第 四 部 分 将 平台 和 多 语言 编程 知识 付 诸 实践 。 从 介绍 Java 7 的 
新 特性 人 手 ， 本 书 曾 盖 了 Java 开发 中 最 重要 的 技术 ， 比 如 依赖 注入 、 而 试 驱 动 的 开发 和 持续 集成 ， 探 索 了 
JVM 上 的 非 Java 语言 ， 并 详细 讲解 了 多 语言 项 目 , 特别 是 涉及 Groovy、Scala 和 Clojure 语言 的 项 目 。 此 外 ， 
书 中 含有 大 量 代 码 示例 ， 帮 助 读者 从 实践 中 理解 Java 语言 和 平台 。 

本 书 适合 Java 开发 人 员 以 及 对 Java7 和 JVM 新 语言 感 兴 趣 的 各 领域 人 十 阅读 ， 
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“Kirk 说 加 油 站 也 卖 啤酒 ,” 这 是 Ben Evans 跟 我 说 的 第 一 句 话 。 他 来 列 里 特 岛 参加 一 个 开放 型 
Java 人 会议。 我 说 我 通 稍 到 加 油 站 就 是 加 油 ， 但 那 边 抛 角 确 实 有 个 店 卖 啤酒 。Ben 看 起 来 对 我 的 回 
答 有 点 儿 失 望 。 我 在 这 个 希腊 小 岛 上 生活 了 5 年 ， 还 从 来 没 在 加 油 站 买 过 啤酒 。 

当 我 在 看 这 本 书 时 ， 那 种 似曾相识 的 感觉 又 来 了 。 我 自 认 为 是 一 名 Java 专 家 : 用 Java 写 了 15 年 程 
序 , 发 表 了 几 百 第 文章 , 在 各 种 会 议 中 演讲 , 还 执教 Java 高 级 课程 。 可 阅读 Ben 和 Martijn 的 这 本 大 作 ， 
经 常 能 给 我 一 些 意料 之 外 的 局 发 。 他 们 一 开始 先 介绍 了 为 改变 Java 生 态 系统 所 做 的 开发 工作 。 类 库 
的 内 部 实现 修改 起 来 相对 容易 ， 一 般 也 能 见 到 成 效 。 例 如 ，Arrays .sort() 的 内 部 实现 在 Java 7 中 
不 再 用 MergeSort 算 法 ， 而 是 改 用 了 Timsort。 由 于 这 个 变化 ， 你 不 用 修改 目 己 对 俩 序数 组 进行 排序 的 
代码 就 可 能 看 到 性 能 的 提升 。 然 而 ， 修 改 类 文件 格式 或 添加 新 的 VM 特 性 则 需要 大 量 工作 。Ben 了解 
这 些 情况 ， 因 为 他 在 JCP 执 行 委 员 会 任职 。 这 本 书 也 是 天 于 Java 7 的 ,所 以 你 能 接触 到 Java 7 中 的 所 有 
新 特性 ， 比 如 语法 糖 的 完善 、string 上 的 switch、 分 文 /合并 ， 还 有 Java NIO.2。 

并 发 就 是 线程 和 同步 ， 对 吗 ?” 如果 这 就 是 你 对 多 线程 的 认识 , 那么 你 需要 学 习 新 知识 了 。 就 
像 作者 在 书 中 指出 的 ,“ 并 发 领域 的 研究 工作 正 开展 得 热火 朝天 "。 与 并 发 相关 的 邮件 列表 上 每 天 
都 有 讨论 ， 新 点 子 层出不穷 。 本 书 会 告诉 你 如 何 看 待 分 而 治之 衬 略 以 及 如 何 规 洲 某 些 安全 隐 阱 。 

在 我 看 到 类 加 载 那 一 章 时 , 我 觉得 他 们 说 得 有 点 儿 过 了 。 那 都 是 我 和 朋友 们 过 去 用 来 炫 砍 的 
技巧 ， 居 然 也 给 摆 出 来 供 大 家 研习 了 1 他 们 讲解 了 javap (这 个 小 工具 用 于 透视 Java 编 译 占 生成 
的 字 节 码 ) 的 工作 方式 ， 还 谈 到 了 新 的 invokedynamic 指 令 ， 并 解释 了 它 跟 普通 反射 的 区 别 。 

我 特别 喜欢 讲 性 能 调 优 的 第 6 章 。 除 了 Jack Shirazi 的 .Java Performance Tuning， 这 还 是 第 一 本 
能 够 抓 住 “如 何 使 系统 运行 更 快 ” 这 个 本 质问 题 的 书 。 我 可 以 用 四 个 字 来 总 结 这 一 章 的 内 容 :“ 测 
量 ， 别 猜 .” 这 是 做 好 性 能 调 优 的 本 质 ， 因 为 人 们 不 可 能 猜 到 运行 慢 的 是 哪 段 代 码 。 这 一 章 从 硬 
件 的 角度 解读 了 性 能 方面 的 问题 ， 而 不 是 只 提供 编码 技巧 。 作 者 还 向 你 展示 了 如 何 测量 性 能 。 有 
一 个 挺 有 意思 的 基础 测试 小 工具 一 cacheTester 类 ， 可 用 于 查看 缓存 未 命中 时 的 开销 。 

本 书 第 三 部 分 介绍 了 JVM 上 的 多 语言 编程 。Java 不 仅仅 是 Java 编 程 语 言 ， 它 还 是 一 个 可 以 运 
行 其 他 语言 的 平台 。 我 们 已 经 见 过 不 同类 型 语言 的 爆炸 式 增 长 了 。 有 些 是 肾 数 式 的 ， 有些 是 声明 
起 的 ， 还 有 一 些 是 平台 的 接口 (Jython 和 JRuby )， 让 其 他 语言 可 以 在 JVM 上 运行 。 语 言 分 为 动态 
的 ( 如 Groovy ) 和 静态 的 ( 如 Javag 和 Scala )。 在 JVM 上 我 们 可 能 因为 多 种 原因 而 使 用 非 Java 的 语 
育 。 如 条 正好 要 开始 一 个 新 项 目 ， 在 做 决定 之 前 先 看 看 都 有 什么 可 用 吧 。 你 可 能 不 用 再 写 那么 多 
套路 化 的 代码 本 。 
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Ben 和 Martijin 向 我 们 介绍 了 三 种 备 选 语言 ，Groovy、Scala 和 Clojure。 在 我 看 来 ,它们 是 当下 
最 切实 可 行 的 选择 。 作 者 描述 了 这 些 语言 之 间 的 差异 、 与 Java 的 差异 以 及 它们 的 特性 。 不 需要 太 
多 的 技术 细节 , 介绍 每 种 语言 的 各 章 足 以 帮 你 弄 清 楚 应 该 用 哪 一 种 。 别 指望 能 在 书 中 看 到 Groovy 
的 参考 手册 ， 但 你 会 了 解 哪 种 语言 更 适合 你 。 

之 后 , 你 将 深信 了 解 如 何 进行 测试 驱动 开发 以 及 如 何 持续 集成 系统 。 我 发 现 一 件 很 有 意思 的 
事 ， 忠 实 的 “ 老 管 家 ”Hudson 这 人 么 快 就 被 Jenkins 取 代 了 。 无 论 如 何 ， 这 些 工 具 跟 Checkstyle 和 
FindBugs 一 样 都 是 项 目 管理 的 基本 工具 。 

你 有 望 通过 研读 本 书 成 为 一 名 优秀 的 Java 开 发 人 人员。 不仅 如 此 ， 你 还 能 了 解 如 何 保持 优秀 。 
Java 一 直 在 变 。 下 一 版 中 我 们 将 见 到 lambda 表 达 式 和 模块 化 。” “人们 也 在 不 断 设 计 新 语言 ， 不 断 
更 新 并 发 结构 。 你 现在 了 解 的 很 多 真相 将 来 可 能 不 再 是 真相 。 因 此 ， 我 们 必须 活 到 老 学 到 老 | 

一 天 ， 我 又 开车 路 过 Ben 想 买 啤酒 的 那个 加 汕 站 。 在 经 济 状 况 如 此 低迷 的 和 希腊 ， 它 也 像 很 多 
公司 一 样 关 张 了 。 我 再 也 不 可 能 知道 他 们 卖 不 卖 啤酒 了 。 


Heinz Kabutz 博 士 
知名 Java 技 术 教 育 家 、The Java Specialists"Newsletter 创 始 人 


DD 实现 Java 模 块 化 的 Jigsaw 项 目 被 延 后 到 Java 9 了 ， 至 少 要 到 2015 年 。 一 一 译 者 注 
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本 书 最 开始 是 给 德意志 银行 外 汇 IT 部 的 新 人 准备 的 培训 笔记 。Ben 觉 得 市 面 上 没有 面 问 经 验 
匮 丢 的 Java 开 发 人 员 的 书 ， 所 以 决定 写 一 本 来 填补 这 个 空 日 。 

在 德意志 银行 IT 管 理 团队 的 支持 下 ，Ben 去 了 比利时 的 Devoxx 会 议 寻 找 灵 感 。 在 那里 他 见 到 
了 IBM 的 三 位 工程 师 ( Rob Nicholson 、Zoe Slattery 和 Holly Cummins ), 他 们 把 他 引荐 给 了 伦敦 Java 
社区 (LIC,， 伦敦 Java 用 户 组 )。 

接 下 来 的 周 六 正好 是 LIC 组 织 的 年 度 开放 会 议 , 就 在 那 次 会 议 上 ，Ben 明 到 了 LIC 的 一 位 领导 
者 一 一 Martijn Verburg。 两 人 一 见 如 故 ， 把 酒 言 欢 ， 悍 悍 相 惜 ， 大 有 相 见 恨 晚 之 意 。 也 正 是 两 人 
对 技术 和 教学 的 共同 热爱 促成 了 本 书 。 

软件 开发 是 一 项 社会 活动 ,我 们 希望 能 借助 本 书 别 啊 这 一 主题 。 我 们 认为 , 虽然 在 这 项 活动 
中 技术 占有 很 重要 的 地 位 , 但 人 与 人 之 间 微 妙 的 沟通 和 交互 关系 也 不 容 忽 视 。 要 在 书 里 轻松 解释 
这 些 东 西 并 不 容易 ， 但 这 一 主题 目 始 至 终 叶 穿 本 书 。 

凭借 着 对 技术 的 执着 和 对 学 习 的 热爱 , 开发 人 员 孜 孜 不 倦 地 工作 着 ,。 我 们 希望 本 书 讨 论 的 一 
些 话题 能 够 激发 他 们 的 学 习 热 情 。 这 是 一 次 观光 之 旅 ， 而 不 是 百科 全 书 式 的 灌输 ， 这 就 是 我 们 的 
初衷 : 帮助 休 人 门 ， 然 后 让 你 日 己 去 探索 那些 激发 你 想象 力 的 东西 。 

本 书 不 仅 为 大 学 毕业 生 准 备 了 接 引 指南 ， 更 为 所 有 心 有 困 惑 的 Java 开 发 人 员 提 供 了 指导 。 
为 他 们 都 很 想 知 道 :“ 接 下 来 我 该 学 什么 ”未 来 要 向 什么 方向 发 展 ? 我 要 青 好 好 考 虚 考 虑 1” 

从 Java 7 的 新 特性 到 现代 软件 开发 的 最 佳 实践 ， 再 到 平台 的 未 来 发 展 ， 本 书 一 路 向 前 ， 向 你 
展示 在 成 长 为 资深 Java 开 发 人 员 的 过 程 中 我 们 认为 至 关 重 要 的 那些 知识 。 并 发 、 性 能 、 字 节 码 和 
类 加 载 是 最 让 我 们 着 迷 的 核心 技术 。 我 们 还 会 谈 到 JVM 上 那些 新 的 非 Java 语 言 ( 即 多 语言 编程 )， 
因为 在 接 下 来 的 几 年 里 ， 对 于 很 多 开发 人 员 来 说 它们 将 变 得 越 来 越 重要 。 

上 归根结底 ,这 是 一 次 以 你 和 你 的 兴趣 为 核心 的 、 具 有 前 胜 性 的 旅程 。 我 们 认为 成 为 一 名 优秀 
的 Java 开 发 人 员 有 助 于 你 彻底 投入 到 工作 中 去 并 顺利 驾驭 开发 ,也 有 助 于 你 对 不 断 变 化 的 Java 世 
界 及 它 的 周边 生态 系统 有 更 多 了 解 。 

我 们 希望 这 本 “经 验 的 结晶 ”对 你 来 说 既 实 用 又 有 趣 , 希望 它 能 让 你 深思 ， 同 时 还 能 带 给 你 
快乐 。 无 论 如 何 ， 写 这 本 书 的 体验 确实 如 此 ! 
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老话 说 得 好 ,“ 人 人 人 拾 柴火 焰 高 ”。 对 于 本 书 来 说 ， 这 人 句 话 非常 贴切 。 如 条 没 有 朋友 、 杀 人 、 
同事 、 同 行 ， 甚 至 偶尔 跟 我 们 对 立 的 那些 人 ， 我们 就 不 可 能 完成 本 书 。 我 们 一 直 都 非常 幸运 ， 
为 那些 对 我 们 批评 最 强烈 的 人 也 可 以 算是 我 们 的 朋友 。 

帮助 过 我 们 的 人 太 多 了 ,很 难 全 部 列举 出 来 ,本 书 快 付 印 的 时 候 , 我 们 在 http://www.java7deve- 
loper.com 上 发 过 一 个 帖子 ， 其 中 也 列 出 了 一 批 名 单 ， 那 些 人 值得 我 们 感谢 。 

如 果 名 单 里 遗漏 了 谁 ， 都 怪我 们 没有 牢记 您 的 大 名 ， 请 接受 我 们 的 烙 意 ! 下 面 这 些 人 对 本 书 
出 版 都 有 贡献 ， 在 此 一 并 感谢 ( 排名 不 分 先后 上 


伦敦 Java 社区 


伦敦 Java 社 区 ( LJC，www.meetup.com/londonjavacommunity ) 是 我 们 两 位 作者 相遇 相知 的 地 
方 。 我们 要 感谢 下 面 这 些 帮 忙 审 校本 书 的 人 : Peter Budo、Nick Harkin、 Jodev Devassy、 Craig Silk、 
N., Vanderwildt,、 Adam J. Markham 、 “Rozallin” 、Daniel Lemon, Frank Appiah、P Franc、 “Sebkom” 
Praveen 、Dinuk Weerasinghe Tim Murray Brown. Luis Murbina、 Richard Doherty 、Rashul Hussan. 
John Stevenson、 Gemma Silvers 、Kevin Wright、Amanda Waite、jJoel Gluth、Richard Paul 、Colin 
Vipurs、Antony Stubbs 、Michael Joyce、Mark Hindess、Nuno 、Jon Poulton 、Adrian Smith 、Ioanmils 
Mavroukakis、Chris Reed 、Martin Skurla、Sandro Mancuso 和 Arml Dhesiaseelan。 

在 Java 语 言 之 外 ， 我 们 得 到 了 James Cook、Alex Anderson 、Leonard Axelsson、Colin Howe、 
Bruce Durling 和 Russel Winder 博 士 非常 认真 的 指导 。 在 这 里 要 特别 感谢 他 们 。 

我 们 还 要 特别 感谢 LJC JCP 委 员 会 成 员 ; Mike Barker、Trisha Gee 、Jim Gough 、Richard 
Warburton 、Simon Maple 、Somay Nakhal 和 David Illsley。 

最 后 ， 感 谢 LJC 的 发 起 人 Barry Cranford。 四 年 前 ,他 市 领 几 个 勇敢 的 人 ， 人 怀抱 一 个 梦想 创建 
了 LIJC。 现 在， LIC 已 经 有 约 2500 名 成 员 ， 并 且 很 多 其 他 技术 社区 也 发 源 于 它 。LJC 已 经 成 为 伦敦 
技术 界 的 中 流 研 柱 。 


www.coderanch.com 
我 们 要 感谢 Maneesh Godbole 、UIlf Ditmer、David O"Meara、Devaka Cooray、Greg Charles、 


Deepak Balu, Fred Rosenberger、 Jesper De Jong、Wouter OQet、 David O'Meara.、 Mark Spritzler 和 和 
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谢 VII 


Roel De Nijs， 因 为 他 们 提供 了 详细 的 评论 和 宝贵 的 反馈 。 
Manning 出 版 社 


感谢 Manning 的 Marjan Bace 接 受 我 们 这 两 个 有 着 疯狂 想法 的 作者 。 在 制作 本 书 出 版 的 整个 过 
程 中 ， 我 们 跟 许 多 人 打 过 交道 ， 非 常 感谢 他 们 : Renae Gregoire、Karen G Miller、Andy Carroll、 
Elizabeth Martin、Mary Piergies 、Dennis Dalinnik 和 Janet Vail。 毫 无 疑问 ， 还 要 感谢 那些 我 们 从 未 
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大 于 本 书 


欢迎 阅读 本 书 。 通 过 阅读 本 书 ， 你 将 成 为 紧 跟 时 代 潮 流 的 Java 程 序 员 ， 重 燃 对 这 一 语言 和 平 
人 台 的 热情 。 学 习 过 程 中 ， 你 会 发 现 Java 7 的 新 特性 ， 熟 悉 重 要 的 现代 软件 技术 〈 比如 依赖 注入 、 
测试 驱动 开发 和 持续 集成 )， 并 开始 探索 JVM 上 的 非 Java 硬 言 这 个 美丽 新 世界 。 

首先 ,我们 来 看 看 James Iry 在 他 精彩 的 博文 “A Brief, Incomplete, and Mostly Wrong History of 
Programming Languages”( 简明 、 不 完整 并 且 泪 润 碧 出 的 编程 语言 历史 ) 中 对 Java 语 言 的 描述 : 

1996 年 ，James Gosling 发 明了 Java。Java 相 对 繁 珊 、 基 于 类 ， 是 支持 垃圾 收集 、 静 

态 类 型 、 单 派发 的 面向 对 象 语言 ， 继 承 方 式 为 实现 单 继承 和 接口 多 继承 。Sun 大 建 宣扬 

Java 的 新 颖 性 。 

他 对 Java 的 描述 基本 上 是 在 插 笠 打 主 ，C# 在 文中 也 受到 了 同等 待遇 但 作为 对 一 种 语言 的 描 
述 , 这 种 方式 也 不 其 。 博文 还 有 很 多 精彩 之 处 , 参见 James 的 博客 ( http://james-iry.blogspot.com/ )。 
没事 的 时 候 看 看 还 是 挺 有 收获 的 。 

James 的 描述 的 确 提出 了 一 个 很 实际 的 问题 。 为 什么 我 们 还 要 讨论 一 种 有 将 近 16 年 历史 的 语 
言 呢 ? 它 真 的 已 经 稳定 ， 没 有 多 少 新 东西 或 有 意思 的 事情 值得 探讨 了 吗 ? 

如 果真 是 那样 ， 本 书 就 会 很 洲 。 事 实 是 ,我 们 依然 在 谈论 Java， 因 为 它 的 一 大 优点 就 是 其 在 
以 下 几 个 核心 设计 决策 之 上 的 构造 能 力 ， 这 些 都 已 经 在 市 场 中 获得 了 了 成功: 

口 运行 时 环境 的 自动 管理 ( 比如 垃圾 收集 、 即 时 编译 ); 

口 语法 简单 ， 核 心 概念 相对 较 少 ; 

口 保守 的 语言 进化 方式 ; 

口 在 类 库 中 增加 功能 和 复杂 性 ; 

口 广泛 、 开 放 的 生态 系统 。 

这 些 设计 决策 一 直 在 推动 着 Java 址 界 的 创新 ， 简 单 的 核心 使 得 开发 门槛 很 低 ， 而 广阔 的 生态 
系统 使 得 后 来 者 很 容易 找到 适合 日 己 需 要 的 现成 组 件 。 

尽管 从 历史 趋势 上 来 看 语言 的 变化 很 缓慢 ,但 这 些 特 质 使 得 Java 平 台 和 语言 既 强 大 又 充满 活 
力 。Java 7 仍然 延续 了 这 一 趋势 。 语 言 的 改变 是 演进 式 ， 而 不 是 革命 式 的 。 然 而 ，Java 7 跟 之 前 版 
本 相 比 有 一 个 主要 区 别 : 它 是 第 一 个 明确 着 眼 于 下 一 次 发 布 的 新 版 本 , 根据 Oracle 有 关 发 布 的 “B 
计划 ”，jJava 7 为 Java 8 的 主要 变化 打下 了 基础 。 

近年 来 ， JVM 上 非 Java 语 言 的 刚 起 也 一 个 重大 变化 。 这 引发 了 Java 和 其 他 JVM 语 言 之 间 的 相 
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XX 关于 本 书 


互 融合 。 现 在 有 大 量 的 项 目 完全 运行 在 JVM 之 上 (这 个 数量 还 在 增加 )， 而 Java 只 是 它们 所 用 的 
编程 请 言 之 一 。 

多 语言 特别 是 涉及 Groovy .Scala 和 Clojure 语 言 的 项 目 ,是 当前 Java 生 态 系 统 的 一 个 重要 因素 ， 
也 是 本 书 最 后 一 部 分 的 主题 。 


阅读 须知 


本 书 内 容 大 体 上 适合 顺序 阅读 , 但 我 们 也 能 理解 某 些 读者 想 直 奔 主 题 的 心情 ,因此 也 在 一 定 
程度 上 迎合 了 这 种 阅读 需求 。 

我 们 非常 认同 自己 动手 的 学 习 方法 ,所 以 建议 读者 在 阅读 的 同时 党 试 示 例 代 码 。 接 下 来 介绍 
本 书 主要 内 容 ， 和 希望 习 惯 跳 夏 阅读 的 读者 能 从 这 里 找到 线索 。 

本 书 分 四 部 分 : 

口 用 Java 7 做 开发 ; 

口 关键 技术 ; 

DJVM 上 的 多 语言 编程 ; 

口 名 语 种 项 目 开 发 。 

第 一 部 分 共 两 章 , 都 是 关于 Java 7 的 内 容 。 本 书 通 篇 使 用 Java 7 的 语法 和 语义 , 所 以 第 1 章 “ 初 
识 Java 7” 是 必 读 的 。 那 些 要 人 处理 文件 、 文 件 系 统 和 网 络 L1O 的 开发 人 员 应 该 会 对 第 2 章 “ 新 VO” 
特别 感 兴趣 ， 

第 二 部 分 共 四 章 ( 第 3-6 章 )， 涉 及 的 主题 包括 依赖 注入、 现代 并 发 、 类 文件 / 字 节 码 以 及 性 
能 调 优 ， 

第 三 部 分 共 四 章 (第 7~10 章 ) 介绍 了 JVM 上 的 多 语言 编程 。 第 7 章 是 必 读 的 ， 因 为 这 一 章 介 
绍 的 JVM 上 可 用 语言 的 类 型 和 使 用 是 阅读 后 面 章 节 的 基础 。 接 下 来 的 三 章 分 别 介绍 与 Java 类 似 的 
语言 Groovy、 兼 具 OO 和 郴 数 式 特 色 的 混 人 台 语 言 Scala 和 纯 枉 数 式 语 言 Clojure。 刚 接触 函数 式 编 程 
的 开发 人 员 可 能 需要 按 顺 序 阅 读 ， 但 这 几 章 本 身 是 相互 独立 的 ， 可 以 跳 着 读 。 

第 四 部 分 ( 最 后 四 章 ) 在 之 前 内 容 基 础 上 介绍 了 新 内 容 。 虽 然 各 章 可 以 独立 阅读 ,但 是 在 某 
些 部 分 我 们 会 假定 你 已 经 读 过 之 前 的 内 容 ， 或 者 已 经 熟悉 那些 主题 。 

简 言 之 ， 如 果 整 本 书 你 必 看 一 章 ， 那 就 看 第 1 章 。 如 果 你 会 看 第 三 部 分 ， 那 一 定 要 看 第 7 章 。 
其 他 各 章 既 可 以 顺序 阅读 ， 也 可 以 独立 阅读 ， 但 后 面 的 某 些 章 届 会 假定 你 已 经 看 过 前 面 的 内 容 。 


读者 对 象 

本 书 主要 是 为 那些 希望 掌握 Java 诸 言 和 平台 现代 化 知识 的 开发 人 员 写 的 。 如 果 你 想 跟 上 Java 
7 的 步伐 ， 就 请 阅读 本 书 吧 。 

如 果 你 扯 提升 一 下 上 自己 的 技能 ， 想 搞 清 楚 依 赖 注 入 、 并 发 、 测 试 驱动 开发 之 类 的 主题 ,本 书 
能 为 你 打下 良好 的 基础 。 
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本 书 也 是 为 已 经 认识 到 多 语言 编程 趋势 并 想 深入 下 去 的 开发 人 员 准 备 的 。 具体 来 说 ， 如 果 你 
相 学 习 函 数 式 编程 ， 那 么 本 书 介 绍 Scala 和 Clojure 的 两 章 会 很 有 帮助。 


和 二 区 [全 


第 一 部 分 只 有 两 章 。 第 1 章 介 绍 了 Java7 及 其 Coin 项 目 ， 该 项 目 包含 很 多 小 巧 高 效 的 特性 。 第 
2 章 全 面 介 绍 了 新 LO API， 包 括 对 文件 系统 API 的 全 面 梳理 ， 还 介绍 了 新 的 异步 IO 能 力 。 

第 二 部 分 分 四 章 介 绍 了 Java 7 的 关键 技术 。 第 3 章 告诉 你 依赖 注入 技术 的 源流 ， 接 着 展示 了 
Java 中 的 标准 解决 方案 Guice 3。 第 4 章 阅 述 在 Java 中 如 何 正确 进行 现代 并 发 开发 。 因 为 硬件 行业 
坚定 地 朝 着 多 核 处 理 器 方向 发 展 , 这 个 话题 再 次 成 为 焦点 。 第 5 章 介 绍 了 JVM 的 类 文件 和 字 节 码 ， 
揭示 了 它们 的 秘密 ,让 你 明白 Java 的 工作 原理 。 第 6 章 讲解 Java 应 用 程序 调 优 的 基础 知识 , 并 讨论 
垃圾 收集 器 等 内 容 。 

第 三 部 分 介绍 JVM 上 的 多 语言 编程 ， 由 四 章 组 成 。 多 语言 编程 的 内 容 从 第 7 章 开 始 ， 这 里 讲 
述 了 多 语言 编程 背景 知识 ， 以 及 使 用 另 一 种 语言 的 恰当 时 机 。 第 8 章 介 绍 了 Groovy- Java 动 态 
编程 的 朋友 。Groovy 突 显 了 语法 相似 的 动态 语言 如 何 大 幅 提 升 Java 开 发 人 员 的 生产 率 。 第 9 意 将 
你 带 入 了 靖 数 式 /OO 混合 的 Scala 世 界 。Scala 是 一 种 强大 精炼 的 语言 。 第 10 章 是 为 Lisp 粉 缘 们 准备 的 。 
Clojure 被 广泛 誉 为 “使 用 得 当 的 Lisp"， 它 全 面 展示 了 JVM 上 函数 式 语言 的 力量 。 

第 四 部 分 以 前 三 部 分 的 内 容 为 基础 , 讨论 多 语言 编程 技术 在 几 个 编程 领域 涉及 的 问题 。 第 
章 谈 到 了 测试 驱动 开发 ， 还 提供 了 一 个 围绕 处 理 模拟 对 象 的 方法 ， 给 出 了 一 些 实战 建议 。 第 12 
董 介 绍 了 两 种 得 到 广泛 应 用 的 工具 ， 用 于 构建 流程 中 的 Maven 3 和 用 于 持续 集成 的 
Jenkins/Hudson。 第 13 章 涵盖 丁 与 快速 Web 开 发 相关 的 主题 ， 解 释 耳 Java 在 这 一 领域 的 传统 缺陷 ， 
并 提供 了 一 些 原型 化 的 新 技术 ( Grails 和 Compojure )。 第 14 章 是 对 全 书 的 总 结 和 对 未 来 的 展望 ， 
其 中 包括 Java 8 可 能 支持 的 新 功能 。 


代码 约定 及 下 载 


首先 需要 下 载 和 安装 的 是 Java7。 只 要 找到 适合 你 的 OS 的 发 布 包 ， 按 照 下 载 和 安装 说 明 来 做 
束 行 了 。Oracle 网 站 上 Java SE 部 分 有 安装 包 的 在 线 下 载 和 说 明 ， 网 址 是 www.oracle.comytechnet- 
work/ Java/Javase/downloads/index.html. 

男 请 参见 附录 A 中 的 安装 说 明 以 及 源码 运行 的 指南 。 

书 中 所 有 源码 都 是 等 览 字 体 ， 以 区 别 于 周围 的 文字 。 很 多 代码 清单 中 都 有 注解 ， 指 出 其 
中 的 关键 概念 ， 有 时 候 文 中 使 用 带 圆 圈 的 数字 ， 对 代码 进行 更 详细 的 注解 。 我 们 尽量 按照 版 
心 的 宽度 设置 代码 格式 ， 也 在 必要 时 换行 ， 并 谨慎 使 用 缩 进 。 但 有 时 候 代 码 行 太 长 ， 因 此 会 
有 续 行 标记 。 

书 中 所 有 示例 的 源码 都 可 在 www.manning.com/TheWell-GroundedJavaDeveloper 找 到 。” 书 中 
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可 | 


山本 书 源 码 也 可 在 图 灵 社 区 http:/wwwituring.com.cn/book/1027 下 载 。 一 一 编者 注 
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XIT 。 关于 本 书 


自始至终 都 有 示例 代码 。 比 较 长 的 代码 清单 有 清楚 的 清单 标题 ,得 一 些 的 代码 清单 就 在 文字 的 行 
与 行 之 间 。 
软件 需求 

如 今 ，Java 7 几乎 可 运行 在 任何 现代 平台 上 。 只 要 你 使 用 以 下 某 个 操作 系统 ， 就 可 以 运行 源 
人 码 示例 : 

口 MS Windows XP 及 以 上 版 本 : 

口 较 新 版 的 *nix; 

口 Mac OS XX 10.6 及 以 上 版 本 。 

多 数 人 都 想 在 IDE 中 试验 代码 示例 。 以 下 这 些 主流 IDE 都 对 Java 7 和 最 新 版 的 Groovy、Scala、 
Clojure 提 供 眼 好 交 持 : 

口 Eclipse 3.7.1 及 以 上 版 本 ; 

口 NetBeans 7.0.1 及 以 上 版 本 : 

口 IntelliJ 10.5.2 及 以 上 版 本 。 

我 们 使 用 了 NetBeans 7.1 和 Eclipse 3.7.1 来 创建 和 运行 代码 示例 。 


作者 在 线 


购买 本 书 英 文 版 的 读者 可 以 免费 访问 由 Manning 出 版 社 运营 的 专用 Web 论 坛 ， 并 在 论坛 中 对 
该 书 进 行 评 论 、 提 出 技术 问题 、 从 作者 和 其 他 用 户 那 里 得 到 帮助 。 要 访问 并 订阅 该 论坛 ， 请 访问 
www.manning.com/TheWell-GroundedJavaDeveloper， 这 个 页 面 介 绍 了 注册 后 如 何 访问 论坛 、 可 以 
得 到 什么 帮助 以 及 论坛 上 的 行为 规则 。 

Manning 问 庶 痢 式 诸 为 读者 之 回 、 谍 者 与 作者 之 间 提 供 一 个 可 以 对 话 的 场所 ， 但 不 会 强制 作 
者 参与 ， 作 者 在 论坛 上 的 页 献 都 是 自 不 而 且 不 收费 的 。 为 使 作者 感 兴 趣 ， 提 高 其 参与 度 ， 我们 建 
议 读者 同 作者 提 一 些 具有 挑战 性 的 问题 。 

只 要 本 书 英文 版 仍然 在 售 , 读者 就 可 以 从 出 版 社 的 网 站 上 访问 作者 在 线 论坛 和 之 前 讨论 话题 
的 归档 。 
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关于 作者 


Ben Evans 是 伦敦 Java 用 户 组 的 发 起 人 、 协 助 定义 Java 生 态 系 统 标 准 的 Java 社 区 过 程 执行 委员 
会 成 员 。 他 在 技术 圈 已 经 度 过 了 很 多 年 “有 趣 的 时 光 "， 现 为 一 家 面向 金融 业 的 Java 技 术 公 司 的 
CEO。Ben 经 常 在 公开 场合 发 表 关 于 Java 平 台 、 性 能 和 并 发 的 演讲 。 

Martijn Verburg ( 是 jClarity 的 CTO ) 作为 一 名 技术 专家 和 众多 初创 企业 的 OSS 守 师 ， 拥 有 十 
多 年 的 经 验 。 他 也 是 伦敦 Java 用 户 组 的 领导 者 , 带领 全 球 的 Java 用 户 组 成 员 为 JSR( 采用 JSR 计 划 ) 
和 OpenJDK ( 采用 OpenJDK 计 划 ) 作出 了 贡献 。 作 为 一 位 公认 的 技术 团队 优化 专家 ， 他 经 党 应邀 
出 席 Java 界 的 大 型 会 议 ( Java0ne、Devoxx、OSCON、FOSDEM 等 ) 并 发 表演 讲 ， 人 送 雅 号 “ 开 
发 魔 头 "”， 先 颂 他 敢于 向 行业 现状 挑战 的 精神 。 
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天 于 封面 图 片 


本 书 封面 上 的 画像 标题 为 “ 卖 花 人 ”"， 摘 自 19 世 纪 法 国 出 版 的 沙 利文 . 马 雷 夏 尔 ( Sylvain 
Maréchal ) 四 卷 本 的 地 域 服饰 风俗 纲要 。 其 中 每 幅 插 图 都 是 手工 精心 绘制 并 上 色 的 。 马 雷 夏 尔 这 
套 书展 示 的 丰富 服饰 ， 令 我 们 强烈 感受 到 200 年 前 乡村 与 城镇 的 巨大 文化 差异 。 不 同 地 域 的 人 山 
水 阻隔 ,言语 不 通 。 无 论 奔走 于 街 埠 , 还 是 驻足 于 乡间 ， 通 过 他 们 的 服饰 ， 一 眼 就 能 看 出 他 们 的 
生活 场所 、 职 业 ， 以 及 生活 境况 。 

时 过 境 迁 ， 书 中 描绘 的 那些 区 域 性 服饰 差异 到 如 今 已 经 不 复 存在 。 即 使 是 不 同 国家 ,都 很 难 
再 看 出 人 们 着 装 的 区 别 ， 再 不 必 说 城镇 和 乡村 了 。 或 许 ,我 们 今天 多 姿 多 彩 的 人 生 , 正 是 从 前 那 
此 文化 差异 的 体现 。 只 不 过 ， 如 今 的 生活 更 加 多 元 ， 而 且 技术 环境 下 的 生活 节奏 也 更 快 了 。 

今 时 今日 ， 计 算 机 图 书 层出不穷 ，Manning 就 以 马 雷 夏 尔 这 套 书 中 多 样 性 的 图 片 ， 来 表达 对 
IT 行业 日 新 月 异 的 发 明 与 创造 的 赞美 。 
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用 Ji 7 做 开发 


本 书 前 两 章 主 要 讨论 Java 7 的 高 明之 处 。 为 便于 读者 理解 下 文 ， 第 1 章 先 介绍 了 一 些 可 提 


发 人 员工 作 效率 的 语法 变化 ， 这 些 变 化 并 不 大 ， 但 效果 都 比较 显著 。 第 1 章 在 这 一 部 分 中 


主要 起 抛砖引玉 的 作用 ， 而 另 一 个 主题 ，Java 中 的 新 IO 才 是 主角 。 


更 轻 
需要 
目 和 


产 率 ， 


还 所 


中 还 


优秀 的 Java 开发 人 员 要 了 解 语言 的 新 特性 。Java 7 中 的 新 特性 可 以 使 开发 人 员 的 工作 变 得 
松 。 但 对 于 这 些 新 变化 ， 光 了 解 语法 是 不 够 的 。 为 了 能 迅速 写 出 高 效 、 安 全 的 代码 ， 你 还 
对 实现 这 些 新 特性 的 原因 和 方式 有 深刻 的 认识 。Java 7 的 变化 可 以 大 致 分 为 两 块 ; Coin 项 
NIO.2. 

第 一 块 是 Coin 项 目 ， 包 括 语 言 层 面 的 一 些小 变化 ， 设 计 它 们 的 初 庙 是 提高 开发 人 员 的 生 
但 又 不 会 对 底层 平台 造成 太 大 影响 。 这 些 变化 包括 : 

口 try-with-resources 结构 (可 以 自动 关闭 资源 )， 

口 switch 中 的 字符 串 ; 

口 对 数字 常量 的 改进 ; 

口 Multi-catch (在 一 个 catch 块 中 声明 多 个 要 捕获 的 异常 ) ; 

口 钻石 语法 《在 处 理 认 型 时 不 用 那么 繁琐 了 )。 

这 些 变化 看 起 来 都 不 大 ， 但 探索 这 些 简 单 的 语法 修改 背后 的 语义 迁移 ， 能 让 你 洞察 Java 语 
Java 平台 之 间 的 差别 。 

第 二 块 变化 是 新 IO (NIO.2) API, 距 Java 原 有 的 文件 系统 支持 相 比 ， 它 具有 压倒 性 优势 ， 
供 了 强大 的 异步 能 力 。 这些 变 化 包括 : 

口 用 于 引用 文件 和 类 文件 实体 的 新 Path 结构 ; 

口 简化 文件 的 创建 、 复 制 、 移 动 和 删除 的 工具 类 Files ; 

口 内 建 的 目录 树 导 航 ; 

口 在 后 台 处 理 大 型 IO 的 将 来 式 和 回调 式 异 步 IO。 

第 一 部 分 结束 时 ， 你 会 很 自然 地 用 Java 7 的 方式 来 思考 问题 和 编写 代码 。 我 们 在 后 续 章 节 
会 用 到 Java 7 中 的 新 特性 ， 所 以 你 还 有 机 会 不 断 温习 这 些 新 知识 ， 
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本 章 内 容 

D Java 既 是 编程 请 言 ， 也 是 平台 
D 且 潜 疾 一 成 能 力 强 好 多 
D try-with-resourcestn 

提升 异常 处 理 能 力 


欢迎 进入 Java 7 的 世界 ! 斗 转 星 移 ， 时 过 境 迁 。 当 尘埃 落 定 ， 我 们 终于 见 到 了 Java 7 的 真 容 。 
虽然 看 起 来 有 点 阳 生 ， 但 它 必 将 市 来 全 新 的 体验 ! 跟随 我 们 经 历 过 这 段 探 索 之 旅 ， 你 将 进入 更 广 
阔 的 世界 ， 发 现 更 多 新 特性 、 更 高 明 的 编程 技巧 ， 并 接触 到 JVM 上 运行 的 更 多 编程 语言 。 

现在 ， 我们 先 来 热 热 时。 虽然 只 是 简单 介绍 ,但 还 是 能 让 你 了 解 Java 7 的 强大 特性 。 我 们 会 
先 解释 Java 语 言 和 平台 的 区 别 ， 因 为 有 时 人 们 会 对 这 两 种 说 法 产生 误解 。 

接着 我 们 会 介绍 Coin 项 目 ， 它 汇聚 了 Java 7 里 短小 精 悍 的 新 特性 。 我 们 会 向 你 展示 Java 平 台 
所 接受 、 吸 纳 和 发 布 的 那些 特性 ， 就 是 它们 构成 了 Java 的 变化 。 在 此 之 后 ， 我 们 会 介绍 Coin 项 目 
中 新 引入 的 6 个 主要 特性 。 

你 会 学 到 新 的 语法 ， 比 如 改进 的 异常 处 理 方 式 (multi-catch ) 以 及 try-with-resources 结 构 ， 借 
此 在 处 理 文件 或 其 他 资源 的 代码 中 躲 开 那些 烦人 的 bug。 读 完 本 章 , 你 将 能 用 全 新 的 方式 编写 Java 
代码 ， 并 整装待发 ， 准 备 好 迎接 更 大 的 挑战 

让 我 们 先 来 探讨 一 下 Java 作 为 语言 和 平台 的 双重 角色 ， 这 是 现代 Java 的 核心 。 这 个 知识 点 将 
贯穿 全 书 ， 是 一 个 必须 掌握 的 基本 要 点 。 


1.1 语言 与 平台 


使 用 Java 之 前 ， 我 们 要 先 弄 清楚 Java 语 言 和 Java 平 台 之 间 的 区 别 。 然 而 ， 有 时 候 不 同 的 作者 
对 语言 和 平台 的 构成 会 有 不 同 的 定义 , 所 以 人 们 有 时 不 太 清 楚 两 者 之 间 的 区 别 , 分 不 清 是 语言 还 
是 平台 提供 了 代码 使 用 的 编程 特性 。 

因为 本 书 的 大 部 分 内 容 都 需要 你 理解 两 者 的 区 别 , 所 以 这 里 需要 说 明 一 下 。 以 下 是 我 们 给 出 
的 定义 。 
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口 Java 语 言 在 “关于 本 书 ”中 ， 我 们 提 到 Java 语 言 是 静态 类 型 、 面 向 对 象 的 博 言 ， 布 望 
你 对 这 种 说 法 已 经 非常 熟悉 了 。 Java 语 言 还 有 一 个 非常 明显 的 特点 , 它 是 ( 或 者 说 应 该 是 ) 
人 类 可 读 的 。 

Java 平 台 ”平台 是 提供 运行 时 环境 的 软件 。Java 虚 拟 机 (JVM ) 负责 把 类 文件 形式 ( 人 
类 不 可 读 ) 的 代码 链接 起 来 并 执行 。JVM 不 能 直接 解释 Java 语 言 的 源 文 件 ， 你 要 先 把 源 文 
件 转换 成 类 文件 。 

Java 作 为 软件 系统 之 所 以 能 成 功 ， 主 要 因为 它 是 一 种 标准 。 也 就 是 说 ， 它 有 规范 文件 描述 它 应 
该 如 何 工 作 。 不 同 的 厂商 或 项 目 组 可 以 据 此 推出 自己 的 实现 , 这 些 不 同 实现 的 工作 方式 在 理论 上 是 
相同 的 。 规 范 虽 然 不 能 保证 这 些 实现 处 理 同 一 任务 时 表现 如 何 ， 但 可 以 保证 处 理 结 果 的 正确 性 。 

控制 Java 系 统 的 规范 有 多 种 ， 其 中 最 重要 的 是 《Java 语 言 规范 /(JLS ) 和 《JVM 规 范 》 
(VMSpec ) 在 Java 7 中 ， 这 两 者 之 间 的 界限 愈 发 清晰 。 实 际 上 ，VMSpec 不 再 引用 JLS 中 的 任何 
内 容 ， 如 果 你 认为 这 是 Java 7 重视 Java 之 外 其 他 语言 的 信和 叶 ， 说 明 你 有 见 微 知 著 的 能 力 ! 希望 你 
能 继续 关注 ， 接 下 来 我 们 会 更 加 深入 地 探讨 这 两 个 规范 之 则 的 差别 。 

提 到 Java 的 双重 角色 ， 你 目 然 想 问 :“ 它 们 两 者 之 间 还 有 什么 关联 吗 ? ”如果 它们 在 Java 7 中 
如 此 泾 泗 分明， 又 是 如 何 共 同形 成 我 们 所 熟悉 的 Java 系 统 的 呢 ? 

连接 Java 语 言 和 平台 之 间 的 纽带 是 统一 的 类 文件 ( 即 .class 文 件 ) 格式 定义 。 认 真 研 究 类 文件 
的 定义 能 让 你 获 益 菲 浅 ， 这 是 优秀 Java 程 序 员 向 伟大 Java 程 序 员 转变 的 一 个 途径 。 图 1-1 展 示 了 产 
生 和 使 用 Java 代 码 的 整个 过 程 。 
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图 1-1 ” Java 源码 被 转换 成 .class 文件 ， 在 JIT 编 译 前 被 加 载 处 理 

如 图 所 示 ，Java 代 码 的 演进 过 程 从 我 们 可 以 看 懂 的 Java 源 码 开 始 ， 然 后 由 javac 编 译 成 .class 
文件 ， 变 成 可 以 加 载 到 JVM 中 的 形式 。 值 得 注意 的 是 , 类 文件 在 加 载 过 程 中 通常 都 会 被 处 理 和 修 
改 。 大 多 数 流 行 框架 ( 特别 是 打 着 “企业 级 ”旗号 的 ) 都 会 在 类 加 载 过 程 中 对 类 进行 改造 。 


问 ， 许 多 开发 人 员 还 会 告诉 你 说 .class 中 的 字 节 码 首先 会 被 JVM 解 释 ， 但 在 稍 后 即时 (JIT ) 编 
译 。 然 而 很 多 人 将 字 节 码 含糊 地 理解 为 “在 某 种 虚构 的 或 简化 的 CPU 上 运行 的 机 器 码 ”。 
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实际 上 , JVM 字 节 码 更 像 是 中 途 的 驿站 , 是 一 种 从 人 类 可 读 的 源码 向 机 器 码 过 渡 的 中 间 状 
态 。 用 编译 原理 术语 讲 ， 字 节 码 实际 上 是 一 种 中 间 语 言 (LIL ) 形态 ,不 是 真正 的 机 器 码 。 也 就 
是 说 , 特 Java 源 码 变 成 字 节 码 的 过 程 不 是 C 或 C++ 程序 员 所 理解 的 那 种 编译 。Java 所 谓 的 编译 器 
javac 也 不 同 于 gcc， 实 际 上 它 人 Java 体 系 中 真正 的 编译 
器 是 JIT， 如 图 1-1 所 示 。 

有 人 说 Java 是 “动态 编译 ”的 ， 他 们 所 说 的 编译 是 指 JIT 的 运行 时 编译 ， 不 是 指 构建 时 创 
建 类 文件 的 过 程 。 

所 以 如 果 被 问 及 “java 是 编译 型 语言 还 是 解释 型 语言 ， 你 可 以 回答 “都 是 。 


希望 我 们 已 经 把 Java 语 言 和 Java 平 台 之 间 的 区 别 解释 清楚 了 。 接 下 来 我 们 进 人 下 一 话题 ， 看 
看 Java 7 中 一 些 语法 上 的 调整 ， 先 从 Coin 项 目 中 的 那些 小 变化 开始 。 


1.2” ”Coin 项 目 : 浓缩 的 都 是 精华 


自 2009 年 1 月 起 ，Coin 便 是 Java 7( 和 Java 8 ) 中 一 个 开源 的 于 项 目 。 本 节 ， 我 们 会 以 Coin 项 
目 中 包含 的 小 变化 为 例 ， 解 释 一 下 Java 诸 言 如何 演 进 以 及 那些 特性 是 如 何 被 选中 的 。 
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| ride Flin pe [二 
创建 Coin 项 目 坪 为 了 反映 Java 语 言 中 的 机 小 变动 。 项 目 rp ye 
小 的 变化 (small change comes as coins )， 而 “套用 一 句 老话 ”( to coin a phrase ) 指 的 是 给 Java 
语言 添 一 个 新 的 表述 方式 。 
在 技术 转子 里 ,这 种 文字 游戏 、 奇 思 妙 想 和 躲 不 掉 的 恐怖 双关 语 随处 可 见 。 你 可 能 已 经 对 
此 习以为常 了 。 


我 们 觉得 解释 语言 “为 什么 要 变 ” 和 “ 变 成 了 什么 ”同样 重要 。 在 开发 Java 7 的 过 程 中 ， 
人 们 对 新 语言 特性 产生 了 很 多 兴趣 ， 但 Java 社 区 有 了 时 并 不 明白 要 按时 实现 这 些 特性 需要 多 大 工 
作 量 。 希望 我 们 在 “为 什么 要 变 ” 这 一 同 题 上 能 够 对 你 有 所 启示 ， 也 希望 能 借 此 消除 一 些 荒诞 
的 说 法 。 如 林 你 对 Java 如 何 发 展 进化 不 感 兴 趣 ， 可 以 直接 阅读 1.3 节 ， 看 看 Java 语 言 发 生 了 哪些 
变化 。 

关于 修改 Java 语 言 有 一 个 工作 量 曲 线 ， 某 些 实现 方式 可 能 要 比 其 他 方式 需要 的 工作 量 少 。 如 
图 1-2 所 示 ， 我 们 尽量 把 不 同 的 修改 方式 及 其 所 需 的 工作 量 呈 现 出 来 ， 以 展现 不 同 复杂 诬 及 相关 
工作 量 的 增长 。 

一 般 来 说 ， 工 作 量 越 少 越 好 。 也 就 是 说 ， 如 果 能 用 类 库 实 现 新 特性 ， 那 就 应 该 用 类 库 。 但 有 
些 特性 用 类 库 或 增加 IDE 功 能 实现 起 来 有 难度 ， 蕉 至 根本 不 可 能 ， 那 就 必须 在 平台 的 更 深层 中 
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[ 具 提 供 的 功能 上 
语法 类 
语言 的 新 特性 
类 文件 格式 的 变化 
VM 的 新 特性 


时 间 及 工作 量 
图 1-2 用 不 同方 式 提供 新 功能 所 需 工 作 莉 的 比较 


下 面 是 一 些 特 性 ( 大 部 分 是 Java 7 中 的 )， 它 们 符合 我 们 为 语言 新 特性 定义 的 复杂 度 。 

口 语法 六 一 一 数字 中 的 下 划 线 (Java 7 ); 

口 新 的 语言 小 特性 一 一 try-with-resources ( Java 7 ); 

口 类 文件 格式 的 变化 一 一 注解 (Java 5 ); 

DJVM 的 新 特性 一 一 动态 调用 (Java7 )。 

它 表 示 这 是 宛 余 的 语法 一 一 在 语言 中 已 经 存在 一 


| 

er | EF 

Ee 0 7 EL, Es 

| | 全 一 he i 局 | 

座 ek 一 pr WB 
se 


“语法 糖 ” 是 描述 一 种 语言 特性 的 短语 。 
种 表示 形式 了 一 一 但 语法 糖 用 起 来 更 便捷 。 

一 般 来 说 ,程序 的 语法 糖 在 编译 处 理 早期 会 从 编译 结果 中 移 除 , 变 为 相同 特性 的 基础 表示 
形式 ， 这 称 为 “去 糖化 "。 

因此 , 语法 糖 是 比较 容易 实现 的 修改 。 它们 通常 不 需要 做 太 多 工作 , 只 需要 修改 编译 器 ( 对 
于 Java 来 说 就 是 javac )。 


Coin 项 目 中 ( 以 及 本 章 余 下 的 内 容 ) 都 是 关于 从 Java 语 法 糖 到 小 的 新 特性 这 个 范围 之 内 的 

Coin 项 目的 最 初 建议 阶段 从 2009 年 2 月 持续 到 3 月 ，coin-dev 的 邮件 列表 上 收 到 了 大 约 70 个 提 
议 ， 堆 括 了 各 种 可 能 的 改进 。 甚 至 有 人 开玩笑 说 要 增加 lolcat 风 格 的 多 行 字符 串 ( 把 标题 得 加 在 
好 笑 或 生气 的 猫咪 图 片上 ， 看 你 怎么 想 了 一 一 http:/icanhascheezburgercony )。 

Coin 项 目 提案 的 评判 规则 很 简单 。 贡 献 者 要 完成 三 项 任务 : 

口 提交 一 份 详细 的 提案 来 描述 修改 ( 本质 上 应 该 是 对 Java 语 言 的 修改 ， 而 不 是 针对 虚拟 机 

的 ); 

口 在 邮件 列表 上 和 针对 提案 进行 开放 式 讨论 ， 能 够 接受 其 他 参与 者 建设 性 的 批评 和 建议 ; 

随时 可 以 提供 一 组 能 够 实现 变化 的 补丁 原型 。 

Coin 项 目 很 好 地 示范 了 语言 和 平台 在 未 来 如 何 演 进 ， 所 有 的 修改 都 会 公开 讨论 , 提供 特性 的 
时 期 原型 ， 并 且 呼 吁 公众 参与 。 
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“什么 是 对 规范 的 小 修改 ” ”现在 也 许 就 是 提出 这 个 问题 的 最 好 时 机 。 我 们 马上 要 讨论 在 JLS 
的 第 14.11 节 中 增加 的 一 个 单词 一 "String"。 你 可 能 再 也 找 不 出 比 这 个 更 小 的 修改 了 ， 可 即便 
是 这 样 一 个 修改 都 会 涉及 规范 中 其 他 几 个 地 方 。 


> 开始 pr 征 汪 和 得 在 2006 征 的 JavsOne 天 会 上 基 a 身 负 的 源码 以 GPLV2 放 可 发 
布 ( pe 一 些 Sun 不 具有 版 权 的 源码 )> 当时 正 值 Java 6 发 布 前 后 ， Whale 7 是 Ja a 在 开源 


Wn 列 | lambda-dev 和 mlvm-dev 等 是 讨论 玉 来 各 种 可 能 特性 的 主要 场所 ， 来 自 
五 湖 四 海 的 开发 人 员 都 可 以 借 此 参与 到 创造 Java 7 的 过 程 中 来 。 实 际 上 ,我 们 泰 与 领导 了 “Adopt 
DpenJDK” 计 划 , 为 新 加 入 OpenJDK 的 开发 人 员 提 供 指导 ， 于 动 疏 进 Java。 如 果 你 起 加 入 我 们 ， 
请 访问 http://java.net/p yea pages/AdoptOpenJDK. 


任何 修改 都 会 产生 影响 ， 我 们 必须 内 Java 语言 的 整体 设计 上 来 追踪 这 些 影响 

任何 修改 都 应 该 严格 执行 下 面 这 些 操作 ( 或 至 少 调研 一 下 ): 

口 更 新 JLS，; 

口 在 源码 编译 器 中 实现 一 个 原型 ; 

口 为 修改 增加 必要 的 类 库 支持 ; 

口 编写 测试 和 示例 ; 

口 更 新 文档 。 

除 此 之 外 ， 如 果 修 改 和 触及 VM 或 者 平台 ， 应 该 ; 

口 更 新 VMSpec; 

口 实现 VM 的 修改 ; 

口 在 类 文件 和 VM 工具 中 增加 支持 ，; 

口 考虑 对 反射 的 影响 ; 

口 考虑 对 厅 列 化 的 影响 ; 

口 想 一 想 对 本 地 代码 组 件 的 影响 ， 比 如 Java 本 地 接口 ( JNI )。 

考虑 到 这 些 修改 对 整体 语言 规范 产生 的 影响 ， 这 可 不 是 一 星 半点 儿 的 工作 。 

如 果 你 要 修改 类 型 系统 , 简直 就 是 自 寻死 路 ,类 型 系统 可 是 个 不 折 不 扣 的 雷 区 。 这 不 是 因为 
Java 的 类 型 系统 不 好 ， 而 是 因为 对 于 拥有 多 种 静态 类 型 系统 的 语言 来 说 ,这 些 类 型 系统 之 间 有 可 
能 会 产生 很 多 交叉 点 。 修 改 它们 很 容易 出 现 意 想不到 的 状况 。 

Coin 项 目 选 的 路 线 非 常 明 智 , 它 建 议 贡 献 者 们 在 修改 提案 中 远离 类 型 系统 。 因 为 对 这 种 修改 
而 言 ， 即 使 最 小 的 变化 都 需要 做 大 量 的 工作 ， 所 以 这 种 做 法 也 是 比较 务实 的 。 

在 简单 介绍 了 Coin 项 目的 背景 之 后 ， 接 下 来 该 看 看 它 包 含 哪 些 特性 了 。 


1 地 
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1.3 Coin 项 目 中 的 修改 


Coin 项 目 主要 给 Java 7 引信 了 6 个 新 特性 ， 它 们 分 别 是 switch 语 句 中 的 string、 数 字 常 量 的 
新 形式 、 改 进 的 异常 处 理 、try-with-resources、 钻 石 语法 ， 还 有 变 参 警告 位 置 的 修改 。 

我 们 会 详细 讲解 Coin 项 目 中 的 这 些 变 化 , 讨论 这 些 新 特性 的 语法 和 含义 , 并 尽 可 能 解释 推出 
这 些 特性 背后 的 动机 。 当 然 ， 我 们 也 不 是 要 把 提案 全 部 有 照搬 过 来 ，coin-dev 邮 件 列表 的 归档 里 有 
完整 的 提案 ， 如 果 你 是 一 个 好 奇 的 语言 设计 师 ， 可 以 去 那里 看 看 ， 还 可 以 和 大 家 讨论 你 的 想法 。 


闲 言 少 叙 ， 开 始 介绍 第 一 个 Java 7 新 特性 


T1321 


SWl1 tch 语 句 中 的 string 值 。 


switch 语句 中 的 String 


switch 语 名 是 一 种 高 效 的 多 路 语句 ， 可 以 省 掉 很 多 繁杂 的 做 套 i 判 断 ， 比 如 像 这 样 : 
public void printDay (int dayOfWeek) | 
(dayOfWeek) 1 


switch 


| 
} 


仁 总 身 忆 
二 
臣 蝇 号 民 
全 总 与 己 
Case 
Lm 
忆 总 映 忆 


def au 


0: 


= 


| 


SySstem,. 
SySstenm. 
SySstem,. 
;: SYStem,. 
: Syatem,. 
: SyBtenm. 
: Svastenm ， 
System.err.println(l"Error!").; 


Gut ， 
Gut . 
Gt 
Out 
GUt 


Printlnl"Tuesday")! 
.Printlnl"Wednesday") ; break,; 
.println("Thureday"}) ; break; 
Dut . 
aut . 


printlni"Sunday"); break; 
printlnl"Monday"); break; 
break:; 


printlnl"Friday"); break, 
printlnl"Saturday"); break:; 
break:; 


在 Java 6 及 之 前 ，case 语 句 中 的 常量 只 能 是 pyte、char、short 和 int (也 可 以 是 对 应 的 封 
装 类 型 Byte,、 Character. short 和 Integer ) 或 枚 举 和 常量 。Java 7 规范 中 增加 了 string， 毕 


况 它 也 是 常 


时 类 型 。 


public void printDay (String dayOfWeek) | 
switch (ldayOfWweek) | 
"Sunday": Syastem.out .println("Dimanche"); break; 
"Monday*": System.out .println(l"Lundi"); break; 
System.oGut .println(l"Mardi"); break:; 


| 
| 


= 
二 
安吉 号 局 
CAaASeE 
Case 
Case 
记忆 有 呈 已 


时 中 


"Tuesday™: 
"Wednesday": 


System.out .println("Mercredi").; 


break: 


"Thureday": Syatem.out .println(l"Jeudi"); break; 
System.out .println("Vendredi"); break.; 


"Friday": 


"Saturday”": 
default: System.out .println'("Error: 
is not a day of the vweek"); break; 


System.out .println("Samedi") ; break:; 


‘4 dayOQfWeek 


除 此 之 外 ，switch 语 句 和 以 前 完全 一 样 。 像 Coin 项 目 中 的 许多 新 特性 一 样 ， 这 不 过 是 一 个 
让 你 更 轻松 的 小 小 改进 。 
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1.3.2 更 强 的 数值 文本 表示 法 


当时 有 几 个 与 整 型 语法 相关 的 提案 ， 最 终 被 选中 的 是 下 面 这 两 个 : 

口 数字 常量 ( 如 基本 类 型 中 的 integer ) 可 以 用 二 进 制 文本 表示 ; 

口 在 整 型 常量 中 可 以 使 用 下 划 线 来 提高 可 读 性 。 

这 两 个 改变 乍 看 起 来 都 不 起 眼 ， 但 它们 确实 解决 了 一 直 困 扰 着 Java 程 序 员 的 一 些小 麻烦 。 

这 两 个 新 特性 对 系统 底层 程序 员 , 就 是 那些 整 天 处 理 原始 网 络 协 议 、 加 密 或 沉迷 于 摆弄 比特 
的 人 们 特别 有 用 。 先 来 看 一 下 二 进 制 文 本 。 

1. 二 进 制 文本 

在 Java 7 之 前 ， 如 果 要 处 理 二 进 制 仁 ， 就 必须 依 助 棘手 ( 又 容易 出 错 ) 的 基础 转换 ， 或 者 调 
用 parsex 方 法 。 比 如 说 ， 如 果 想 让 int x 用 位 模式 表示 十 进 制 值 102， 你 可 以 这 样 写 : 

int 其 = Inteaer.parseInt ("1100110", 2);，} 

为 了 确保 x 是 正确 的 位 模式 ， 你 需要 融 许 多 代码 。 这 种 方式 尽管 看 起 来 还 行 ， 但 实际 上 存在 
很 多 问题 ， 

口 十 分 繁琐 ; 

口 方法 调用 对 性 能 有 影响 ; 

口 需要 知道 parseInt |() 的 双 参 形式 ; 

口 需要 记 住 双 参 的 parseInt1() 的 处 理 细节 ; 

口 JIT 强 伴 带 更 难 实 现 ，; 

口 用 运行 时 的 表达 式 表 示 应 该 在 继 译 时 确定 的 常量 ， 守 致 x 不 能 用 在 switch 请 名 中 ; 

口 如 果 在 位 模式 中 有 拼写 错误 ( 能 通过 编译 )， 会 在 运行 时 抛 出 RuntimeException 。 

现在 好 了 ， 用 Java 7 可 以 写成 : 

int x = 0Pl0O01IO; 

我 们 没 说 这 种 方法 无 所 不 能 ， 但 它 确实 解决 了 上 面 提 到 的 那些 问题 。 

你 在 跟 二 进 制 打 交道 时 , 这 个 小 特性 会 是 你 的 得 力 助手 。 比 如 在 处 理 字 节 时 , 可 以 在 switch 
语句 中 使 用 由 位 模式 定义 的 二 进 制 常量 。 

另外 一 个 新 特性 虽然 小 , 但 却 很 实用 
人 和 下划线 。 

2. 数字 中 的 下 划 线 

众所周知 ， 人 脑 和 电脑 有 很 多 不 同 的 地 方 ， 对 于 数字 的 处 理 方式 就 是 其 中 之 一 。 通常 人 们 都 
不 太 高 欢 面 对 一 大 串 数 字 。 这 也 是 我 们 发 明 十 进 制 的 原因 之 一 一 一 因为 人 脑 更 擅 于 处 理 信 息 量 大 
的 短 字 串 ， 而 不 是 每 个 字符 信息 量 都 不 太 多 的 长 字 串 。 

也 就 是 说 ， 我 们 觉得 1c372ba3 要 比 00011100001101110010101110100011 更 容易 处 理 ， 
但 电脑 只 认 第 二 种 。 信 们 在 处 理 长 串 数 字 时 会 采用 分 唱法 ， 比 如 用 404-555-0122 表 示 电 话 号 码 。 


可 以 在 表示 一 组 二 进 制 位 或 其 他 长 数值 的 数字 中 加 
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注意 “如果 你 跟 作 者 ( 欧洲 人 ) 一 样 , 想 知道 为 什么 美国 电影 或 书 里 的 电话 号 码 总 是 以 555 开 头 ， 塌 
我 可 以 告诉 你 。555-01xx 是 保留 号 段 ， 用 于 诬 构 的 情境 。 这 是 为 了 避免 现实 生活 中 的 人 
接 到 那些 对 好 菜 坞 电影 过 分 投入 的 人 打 来 的 电话 。 


其 他 带 有 分 隔 符 的 一 长 申 数字 ， 

口 100 000 000 美 元 ( 一 大 笔 钱 ); 

口 08-92-96 ( 英国 很 行 的 排序 代码 )。 

可 在 代码 中 处 理 数 字 时 不 能 用 逗号 ( ,) 和 连 字 符 ( - ) 作 分 隔 符 ， 因 为 它们 可 能 会 引发 歧义 。 
Coin 项 目 中 的 提案 借用 了 Ruby 的 创意 ， 用 下 划 线 (_) 作 分 隔 符 。 注 意 ， 这 只 是 为 了 让 你 阅读 数 
字 时 更 容易 理解 而 做 的 一 个 小 修改 ， 编 详 器 会 在 编译 时 把 这 些 下 划 线 去 掉 ， 吕 保留 原始 数字 。 

也 就 是 说 ， 为 了 不 把 100 000 和 10 000 000 摘 混 ， 你 可 以 在 代码 中 将 100 000 000 写 成 
ed td be eds si en mim aot 


| anoctherLong = 2 147 483 648L; 
int bitPattern = 0b6001 1100 0011 0111 0010 1011 1010 0011; 


注意 ; 赋 给 anotherLong 的 数值 现在 看 起 来 清楚 多 了 了。 


警告 ”在 Java 中 可 以 用 小 写字 母 1 表 示 长 整 型 数值 ， 比 如 10101001。 但 最 好 还 是 用 大 写字 母 L， 
人 19101090 和 治本 娄 清 划 生 站。 


现在 你 应 该 清 想 这 些 变 化 给 整数 处 理 市 来 的 好 处 了 ! 让 我 们 继续 前 进 ， 去 看 看 Java 7 中 的 异 
党 处 理 。 


1.3.3 ”改善 后 的 异常 处 理 


异常 处 理 有 两 处 改进 一 一 multicatch 和 final 重 抛 。 要 知道 它们 对 我 们 有 什么 帮助 ， 请 先 
看 一 段 Java 6 代码 。 下 面 这 段 代码 试图 查找 、 打 开 、 分 析 配 置 文件 并 处 理 此 过 程 中 可 能 出 现 的 各 
种 异 第 : 


代码 清单 1-1 在 Java 6 中 处 理 不 同 的 异常 
public Contiguraticn getConfig(string fileName}) { 
Conf igquration cfg = null; 
try { 
string fileText = getFile lfileName); 
cfg = Vverifyconfigl(lparseConfig(lfileText)); 
catch (FileNotFoundException fnfx) | 
System.err.println(l"Config file '" + fileName + "' ig missing"); 
catch (IOException e) I 
System.err.println("Error while processing file '" + fileName + ™'™); 
catch (ConfigurationException e) | 
System.err.println(l"Config file '" + fileNMame + "' is not consistent"): 


iss me ae 


1 于 
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} eateh (ParseException e) | 
System.err.println("Config file '" + fileName + " 


} 

return cfg; 

这 个 方法 会 遇 到 的 下 面 几 种 异常 : 

口 配置 文件 不 存在 ; 

口 配置 文件 在 正 要 读 取 时 消失 了 ; 

D 配置 文件 中 有 语法 错误 ; 

口 配置 文件 中 可 能 包含 无 效 信息 。 

这 些 异 常 可 以 分 为 两 大 类 。 一 类 是 文件 以 某 种 方式 丢失 或 损坏 , 另 一 类 是 虽然 文件 理论 上 存 
在 并 且 是 正确 的 ， 却 无 法 正常 读 取 ( 可 能 是 因为 网 络 或 硬件 故障 )。 

如 果 能 把 这 些 异 常情 况 简 化 为 这 两 类 ,并 且 把 所 有 “文件 以 某 种 方式 丢失 或 损坏 ”的 异常 放 
在 一 个 catch 语 名 中 处 理会 更 好 。 在 Java 7 中 就 可 以 做 到 ， 


Public Configuration getConfiglstring fileName) | 

Conf iguration cfg = null; 

try 1 
string fileText = getFile (ftileName); 
cfg = verifyconfig(parseConfig (fileText)).,; 
catch (FileNotFoundException|ParseException|ConfigurationException el | 
Syatem,.err.println("Config file '" + fileName + 

"1 i199 missing or mlformed"),; 

catch (IOException iox) | 
Syatem.err.println('"Error while processing file '" + fileName + 


! ig malformed"): 


ss 


Se 


人 
} 
return cfg; 
} 
异常 e 的 确切 类 型 在 编译 时 还 无 法 得 知 。 这 意味 着 在 cat ch 块 中 只 能 把 它 当 做 可 能 异常 的 共 
同 父 类 ( 在 实际 编码 时 经 常用 Exception 或 Throwable ) 来 处 理 。 
男 外 一 个 新 语法 可 以 为 重新 抛 出 异常 提供 帮助 。 开发 人 员 经 常 要 在 重新 抛 出 异常 之 前 对 它 进 
行 处 理 。 在 前 几 个 版 本 的 Java 中 ， 经 常 可 以 看 到 下 面 这 种 代码 : 
RE 
doSomethingElseWhichMightThrowSoQLException!(); 
} catch (Exception ee 1 


Li @ 
| 
这 会 强迫 你 把 新 扫 出 的 异常 声明 为 Excepcion 类 型 一 一 异常 的 真实 类 型 却 被 覆 普 了 。 
不 管 怎样 , 很 容易 看 出 来 异常 只 能 是 IOException 或 SOLException。 既然 你 能 看 出 来 ， 编 
译 器 当然 也 能 。 下 面 的 代码 中 用 了 Java 7 的 语法 ， 只 改 了 一 个 单词 
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try 1 

doSomethingWhichMightThrowIOException(); 
doSomethingElseWhichMightThrowSQLException(); 
catch (final Exception e) | 


a 


throw ee; 

) 

关键 字 final 表 明 实 际 抛 出 的 异常 就 是 运行 时 过 到 的 异常 一 一 在 上 面 的 代码 中 就 是 
IOException 或 SQLException。 这 被 称 为 final 重 抛 ， 这样 就 不 会 抛 出 第 统 的 异常 类 型 ， 从 而 
屠 免 在 上 层 只 能 用 息 统 的 catch 捕 获 。 

上 例 中 的 关键 字 final 不 是 必需 的 ,但 实际 上 ， 在 癌 catch 和 重 抛 语 义 调整 的 过 渡 阶 段 ， 留 
着 它 可 以 给 你 提 个 醒 。 

Java 7 对 异常 处 理 的 改进 不 仅 限 于 这 些 通用 问题 ， 对 于 特定 的 资源 管理 也 有 所 提升 ， 我 们 马 
上 就 会 讲 到 。 


1.3.4 try-with-resources (TWR) 


这 个 修改 说 起 来 容易 , 但 其 实 暗 藏 玄机 ， 最终 证 明 做 起 来 比 最 初 预想 的 要 难 。 其 基本 设想 是 
把 资源 ( 比如 文件 或 类 似 的 东西 ) 的 作用 域 限定 在 代码 块 内 ， 当 程序 离开 这 个 代码 块 时 ， 资 源 会 
被 自动 关闭 。 

这 是 一 项 非常 重要 的 改进 ， 因 为 没 人 能 在 手动 关闭 资源 时 做 到 100% 正 确 , 甚至 不 久 前 Sun 提 
供 的 操作 指南 都 是 错 的 。 在 向 Coin 项 目 提交 这 一 提案 时 , 提交 者 宣称 JDK 中 有 三 分 之 二 的 close() 
用 法 都 有 bug， 真 是 不 可 思议 1 

好 在 编译 器 可 以 生成 这 种 学 究 化 、 公 式 化 且 手 工 编写 易 犯 错 的 代码 ， 所 以 Java 7 借助 了 编译 

这 可 以 减少 我 们 编写 错误 代码 的 几率 。 相 比 之 下 ， 想 想 你 用 Java 6 写 段 代码 ， 要 从 一 个 URL 
(url ) 中 读 取 字 节 流 ， 并 把 读 取 到 的 内 容 写 人 到 文件 ( out ) 中 , 这 么 做 很 容易 产生 错误 。 代 码 
清单 1-3 是 可 行 方案 之 一 。 


代码 清单 1-3 Java 6 中 的 资源 管理 语法 
Inputstream is = null; 
try { 
is = Url.openSstreaml(); 
QutputStream out = new FileOQutputSstream(file); 
try | 
byte[] buf = new byte[4096]; 
int len:; 


while ((len = is.read(buf)}) sa 0) 由 理 异 常 (能 
out write (buf, 0, len}; 读 或 写 ) 
catch (IOException iox) | : 


i 


finally I 
try 1 


out .close (); 
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} catch (IOException closeOQOutx) 1{ 


} 


} cateh (FileNotFoundBException fnfx) | | 处 理 异常 
} catch (IOException openx) | eg 


} finally 1 
try 1 
if (lis !=s null) is.close!(); 
| catch (IOException closeInx) | 


) 

看 明白 了 吗 ? 重点 是 在 处 理 外 部 资源 时 ,， 墨 菲 定律 ( 任何 事 都 可 能 出 错 ) 一 定 会 生效 ， 比 如 : 

口 URL 中 的 Inputstream 无 法 打开 ， 不 能 读 取 或 无 法 正常 关闭 ; 

口 outputStzeam 对 应 的 File 无 法 打开 ， 无 法 写 人 或 不 能 正常 关闭 ; 

口 上面 的 问题 同时 出 现 。 

最 后 一 种 情况 是 最 让 人 头疼 的 一 一 异常 的 各 种 组 合 拳 打 出 来 令 人 难以 招架 。 

新 语法 能 大 大 减少 错误 发 生 的 可 能 性 , 这 正 是 它 受 欢迎 的 主要 原因 。 编 译 器 不 会 犯 开发 人 员 
编写 代码 时 易 犯 的 错误 。 

让 我 们 看 看 代码 清单 1-3 中 的 代码 用 Java 7 写 出 来 什么 样 。 和 前 面 一 样 ，ur1 是 一 个 指向 下 载 
目标 文件 的 URL 对 象 ，file 是 一 个 保存 下 载 数据 的 File 对 象 。 


代码 清单 1-4 Java 7 中 的 资源 管理 语法 “ 
try (OutputSstream Out = new FileQutputstreaml(lfile), 
InputStream isg = url.openStream() ) 1 
bytel] buf = new byte[l4096]:; 
se = ds.read(buf)) > 0) { 
out .write {lbuf, 0, len): 
| 
| 
这 是 资源 上 自动 化 管理 代码 块 的 基本 形式 一 一 把 资源 放 在 try 的 圆 括 号 内 。C# 程 序 员 看 到 这 个 
也 许 会 觉得 有 点 眼熟 , 是 的 , 它 的 确 很 像 C# 中 的 从 句 , 带 着 这 种 理解 使 用 这 个 新 特性 是 个 不 错 的 
起 点 。 在 这 段 代码 块 中 使 用 的 资源 在 处 理 完成 后 会 自动 关闭 。 
但 使 用 ay-with-resources 特 性 时 还 是 要 小 心 ， 因 为 在 某 些 情况 下 资源 可 能 无 法 关闭 。 比 如 在 下 
面 的 代码 中 , 如 果 从 文件 ( someFile.bin ) 创 建 objeccInputStream 时 出 错 , FileInputStream 


可 能 就 无 法 正确 关闭 。 
try ( ObjectInputStream in = new ObjectInputStream (new 
FileInputStream("someFile.bin"})) ) 1 


es 
假定 文件 ( someFile .bin ) 存在 , 但 可 能 不 是 objectInput 类 型 的 文件 ， 所 以 文件 无 法 正 
确 打 开 。 因 此 不 能 构建 0objectInputstream， 所 以 FileInputstream 也 没 办 法 关闭 。 
要 确保 try-with-resources 和 生效， 正确 的 用 法 是 为 各 个 资源 声明 独立 变量 。 
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try ( FlileInPUESCream fin = new FileInputStream("someFile.bin"):; 
objectInputstream in = new ObjectInputstreaml(lfin) ) | 


上 

TWR 的 男 一 个 好 处 是 改善 了 错误 跟踪 的 能 力 ， 能 够 更 准确 地 跟踪 堆栈 中 的 异常 。 在 Java 7 之 
前 ， 处 理 资源 时 抛 出 的 异常 信息 经 常会 被 覆盖 。TWR 中 可 能 也 会 出 现 这 种 情况 ， 因 此 Java 7 对 跟 
踪 堆 栈 进行 了 改进 ， 现 在 开发 人 员 能 看 到 可 能 会 丢失 的 异常 类 型 信息 。 

比如 在 下 面 这 段 代码 中 ， 有 一 个 返回 InputStream 的 值 为 null 的 方法 ， 


cryfInpucScream i = getNullstream()) { 
i .available'(}; 


} 
在 改进 后 的 跟 踊 堆栈 中 能 看 到 提示 ， 注意 其 中 被 抑制 的 NullPointerException ( 简称 
NPE ). 


Exception in thread "main" java.lang.NullPointerException 
at wgjd.ch0l .ScratchSuprExcep.run(lScratchSuprExcep. java:23) 
at wgjd.ch0l.ScratchSsuprExcep .main(lScratchSuprExcep.Jjava:39) 
Suppresgsed; java. lang.NullPointerExcept ion 
at wgjd.chol.sScratchSuprExcep .runlScratchSuprExcep. java:24) 


1 more 


目 We ER 


源 类 都 必须 实现 这 个 接口 。Java 7 平台 中 的 大 多 数 资 源 类 都 被 修改 过 ， 已 经 实现 了 
AutoCloseable ( Java 7 中 还 定义 了 其 父 接口 Closeable )， 但 并 不 是 全 部 资源 相关 的 类 都 采 


用 了 这 项 新 技术 。 不 过 ， JDBC 4.1 已 经 具备 了 这 个 特性 。 
然而 在 你 自己 的 代码 里 ， 在 需要 处 理 资源 时 一 定 要 用 TWR， 从 而 避免 在 异常 处 理 时 出 现 


bug。 
希望 你 能 尽快 使 用 try-with-resources， 把 那些 多 余 的 bug 从 代码 库 中 赶 走 。 


1.3.5 ”钻石 语法 

针对 创建 泛 型 定义 和 实例 太 过 繁琐 的 问题 ，Java 7 做 了 一 项 改进 ， 以 减少 处 理 泛 型 时 敲 键盘 
的 次 数 。 比 如 你 用 useria ( 整 型 值 ) 标识 一 些 user 对 象 ， 每 个 user 都 对 应 一 个 或 多 个 查找 表 ?。 
这 用 代码 应 该 如 何 表示 呢 ? 


Map<Integer, Map<String, String>»s usersLists = 
new HashMap<lInteger, Map<String, String»>>{(); 


这 简 古 太 长 了 ， 并 且 几 乎 一 半 字 符 都 是 重复 的 。 如 果 能 写成 


山 一 种 为 提高 处 理 速度 而 用 查询 取代 计算 的 处 理 机 制 。 一 般 是 将 事先 计算 好 的 结果 存在 数组 或 映射 中 ， 人 然后 在 需要 
该 结果 时 直接 读 取 ， 比 如 用 三 角 表 查 某 一 角度 的 正弦 值 。 一 一 译 者 注 
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Map<Integer, Map<String, String>> usersLists = new HashMap<>(l : 

让 编译 器 推断 出 右 侧 的 类 型 信息 是 不 是 更 好 ?神奇 的 Coin 项 目 满足 了 你 这 个 心愿 。 在 Java 7 
中 , 像 这 样 的 声明 缩写 完全 全 法， 还 可 以 向 后 兼容 ， 所 以 当 你 需要 处 理 以 前 的 代码 时 ， 可 以 把 过 
去 比较 式 琐 的 声明 去 择 ， 使 用 新 的 类 型 推断 语法 ， 这 样 可 以 省 出 点 儿 空间 来 。 

编 主 辣 为 这 个 特性 有 来 用 了 新 的 类 型 推断 形式 。 它 能 推断 出 表达 式 右 侧 的 正确 类 型 ， 而 不 是 仅 
仅 符 换 成 定义 完整 类 型 的 文本 。 


1 


4 妥当 


下 二 和 本 9 Be = 于 大 -x | 和 Fe 过 | = 二 -| > 而 症 有 ir 而 全 
| 全] 由 PA or rN i te i | 了 
本 可 ue 7 | We . i ' = 四， 村 | Bs 二 人 下 可 本 
ee Ml We i 上 有 村 0 | 
| eee ms et tt ee | EL Ee - 上 
了 是 /过 人 ee > 古人 直 汪 2 Sh 和 pe 四 


”把 它 多 为 ”钻石 语法 ”是 因 人 类 信息 和 直人 人 大 要 中 的 名字 是 为 杠 
实例 创建 而 做 的 类 型 推断 改进 ”( Improved Type Inference for Generic Instance Creati 
名 字 太 长， 可 缩写 ITIGIC 听 上 去 又 很 傻 ， 所 以 干脆 就 叫 钻石 语法 了 。 


新 的 钻石 语法 肯定 会 让 你 少 写 些 代码 ,我 们 最 后 还 要 探讨 Coin 项 目 中 的 一 个 特性 一 使 用 变 
参 时 的 警告 信息 。 


1.3.6 简化 变 参 方法 调用 


这 和 十 了 所 有 修改 里 最 简单 的 一 个 , 只 是 去 挥 了 方法 签名 中 同时 出 现 变 参 和 泛 型 时 才 会 出 现 的 类 
型 索 告 信息 。 

换 句 语 说 ， 除 非 你 写 代 码 时 习惯 使 用 类 型 为 ?的 不 定数 量 参数 ， 并 且 要 用 它们 创建 集合 ， 否 
则 你 就 可 以 进入 下 一 节 了 。 如 果 你 想 要 写 下 面 这 种 代码 ， 那 就 继续 阅读 本 节 : 


public setatic <T> CollectioncT> doSomething[T... entries) { 


| 

还 在 ”很 好 。 这 到 底 是 怎么 回 事 ? 

变 参 方法 是 指 参 数列 表 未 尾 是 数量 不 定 但 类 型 相同 的 参数 方法 。 但 你 可 能 还 不 知道 变 参 方法 
是 如 何 实现 的 。 基 本 上 ， 所 有 出 现在 末尾 的 变 参 都 会 被 放 到 一 个 数组 中 ( 由 编译 器 自动 创建 )， 
并 作为 一 个 参数 传人 。 

这 是 个 好 主意 , 但 是 存在 一 个 公认 的 Java 泛 型 缺陷 一 一 不 允许 创建 已 知 类 型 的 泛 型 数组 。 比 
如 下 面 这 段 代码 ， 编 译 就 无 法 通过 : 

HaghMap<String, String>[] arrayHm = new HashMap<»> [2]; 

不 可 以 创建 特定 证 型 的 数组 ， 只 能 这 样 写 : 

HashMap<String, String>[] warnHm = new HashMap [21] | 

可 这 梓 编 译 带 会 给 出 一 个 只 能 忽略 的 警告 。 你 可 以 将 warnHm 的 类 型 定义 为 HashMap<Strin g, 
string> 数 组 , 但 不 能 创建 这 个 类 型 的 实例 ， 所 以 你 不 得 不 硬 着 头皮 ( 或 至 少 忘掉 警告 ) 硬 生 生 
地 把 原始 类 型 ( HashMap 数 组 ) 的 实例 塞 给 warnHm。 
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这 两 个 特性 ( 编译 时 生成 数组 的 变 参 方法 和 已 知 泛 型 数组 不 能 是 可 实例 化 类 型 ) 碰 到 一 起 时 ， 
会 令 人 有 点 头疼 。 看 看 下 面 这 段 代码 : 


HashMap<String, String> hml = new HashMap<>|(),， 
HashMap<String, Strings hm2 = new HashMapes(), 


Collection<HashMap<String, Strings» coll = doSomething{(hml, hm2); 

编译 器 会 尝试 创建 一 个 包含 hml 和 hm2 的 数组 , 但 这 种 类 型 的 数组 应 该 是 被 严格 禁止 使 用 的 。 
面 对 这 种 进退 两 难 的 局 面 , 编译 带 只 好 违心 地 创建 一 个 本 来 不 应 出 现 的 泛 型 数组 实例 , 但 它 又 觉 
得 目 己 不 能 保持 沉默 ， 所 以 还 得 嘟 只 着 警告 你 这 是 “未 经 检查 或 不 安全 的 操作 ”。 

从 类 型 系统 的 条 度 看 ， 这 非常 合理 。 但 可 怜 的 开发 人 员 本 想 使 用 一 个 十 分 团 谱 的 API， 一 看 
到 这 些 吓 人 的 警告 ， 却 得 不 到 任何 解释 ， 不 免 会 内 心志 正 。 

1. Java 7 中 的 警告 去 了 哪里 

Java 7 的 这 个 新 特性 改变 了 和 警告 的 对 象 。 构 建 这 些 类 型 毕竟 有 破坏 类 型 安全 的 风险 ， 这 总 得 
有 人 知道 。 但 API 的 用 户 对 此 是 无 能 为 力 的 ， 不 管 aosomething() 是 不 是 干 了 坏事 ， 破 坏 了 类 
型 安全 ， 都 不 在 API 用 户 的 控制 范围 之 内 。 

真正 需要 看 到 这 个 警告 信息 的 是 写 dosomething |() 的 人 ， 即 API 的 创建 者 ， 而 不 是 使 用 者 。 
所 以 Java 7 把 警告 信息 从 使 用 API 的 地 方 挪 到 了 定义 API 的 地 方 。 

过 去 是 在 编译 使 用 API 的 代码 时 触发 绚 告 ， 而 现在 是 在 编译 这 种 可 能 会 破坏 类 型 安全 的 API 
时 触发 。 编 译 器 会 警告 创建 这 种 API 的 程序 员 ， 让 他 注意 类 型 系统 的 安全 。 

为 了 减轻 API 开 发 人 员 的 负担 , Java 7 还 提供 了 一 个 新 注解 java .lang .SafeVarargs。 把 这 
个 注解 应 用 到 API 方 法 或 构造 方法 之 中 ， 则 会 产生 类 型 警告 。 通 过 用 esafevarargs 对 这 种 方法 
进行 注解 , 开发 人 员 就 不 会 在 里 面 进 行 任何 危险 的 操作 ， 在 这 种 情况 下 ， 编 译 器 就 不 会 再 发 出 警 
告 了 了 。 

2. 类 型 系统 的 修改 

虽然 把 警告 信息 从 一 个 地 方 挪 到 为 一 个 地 方 不 是 改变 游戏 规则 的 语言 特性 , 但 也 证 明了 我 们 
之 前 提 到 的 观点 一 一 Coin 项 目 曾 奉劝 诸位 贡献 者 远离 类 型 系统 , 因为 把 这 么 一 个 小 变化 讲 清 楚 要 
大 费 周章。 这 个 例子 表明 搞 清 楚 类 型 系统 不 同 特性 之 间 如 何 交 互 是 多 么 费心 费力 , 而 且 对 语言 的 
修改 被 实现 后 又 会 起 么 影响 这 种 交互 。 这 还 不 是 特别 复 玉 的 修改 , 更 大 的 变动 所 涉及 的 内 容 还 会 
更 多 ， 其 中 还 包括 大 量 微妙 的 分 文 。 

最 后 这 个 例子 兰 明 了 由 小 变化 引发 的 错综复杂 的 影响 。 我 们 对 Coin 项 目 中 改进 的 讨论 也 结束 
了 。 尽 管 它 们 几乎 全 都 是 语法 上 的 小 变化 , 但 跟 实 现 它们 的 代码 量 相 比 ， 它 们 所 带 来 的 正面 影响 
还 是 很 可 观 的 。 一 旦 开始 使 用 ， 你 就 会 发 现 这 些 特 性 对 程序 琶 的 很 有 帮助 1 


1.4 小结 


修改 语言 非常 困难 。 而 用 类 库 实 现 新 特性 总 是 相对 容易 一 些 ， 当然 并 不 是 所 有 特性 都 能 用 类 
库 实现 。 面 对 挑战 时 ， 语 言 设计 师 可 能 会 做 出 一 些 比 他 们 的 预想 更 轻微 、 更 保守 的 调整 。 
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现在 , 我 们 该 去 看 看 构成 发 布 版 本 更 重要 的 东西 了 , 先 从 Java 7 中 革 些 核心 类 库 的 变化 开始 。 
我 们 的 下 一 站 是 LO 类 库 ， 那 里 可 以 说 是 发 生 了 天 翻 地 覆 的 变化 。 在 此 之 前 ， 和 希望 你 已 经 掌握 了 
Java 之 前 的 版 本 处 理 VO 的 方法 , 因为 Java7 中 的 这 些 类 ( 有 时 候 被 称 为 NIO.2 ) 是 构建 在 之 前 框架 
基础 之 上 的 。 

如 果 你 想 看 到 更 多 关于 TWR 实 战 的 例子 ， 或 者 想 要 了 解 最 新 、 高 性 能 的 VO 类 ， 那 就 赶快 进 
人 下 一 章 吧 1 
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本 章 内 容 

口 Java 7 的 新 L/O API ( 即 NIO.2 ) 

口 Path 一 一 基于 文件 和 目录 的 IO 新 基础 
口 Files 应 用 类 及 它 的 各 种 辅助 方法 

口 如 何 实 现 常 见 的 IO 应 用 场景 
口 介绍 异步 IO 


本 章 重 点 是 Java 语 言 中 改变 较 大 的 VO API, 被 称 为 “再 次 更 新 的 VO” 或 NIO.2 ( 即 JSR-203 ), 
NIO.2 是 一 组 新 的 类 和 方法 ， 主 要 存在 于 java .nio 包 内 。 下 面 来 看 一 下 它 的 优点 。 

口 它 完全 取代 了 java.io.File 写 文件 系统 的 交互 ， 

口 它 提供 了 新 的 异步 处 理 类 ， 让 你 无 需 手动 配置 线程 地 和 其 他 底层 并 发 控制 ， 便 可 在 后 台 

线程 中 执行 文件 和 网 络 IO 操作 。 

口 它 引 人 了 新 的 Network-channe1 构 造 方 法 ， 简 化 了 套 接 字 ( Socket ) 与 通道 的 编码 工作 。 

先 看 案例 。 老 板 让 你 写 个 程序 ， 要 扫描 生产 服务 器 上 的 所 有 目录 ， 找 出 曾经 用 各 种 读 / 瑟 和 
所 有 者 权限 写 人 过 的 所 有 properties 文 件 。 对 于 Java 6( 及 更 低 版 本 ) 而 言 ， 这 几乎 是 不 可 能 完成 
的 任务 ， 因 为 : 

口 没有 直接 支持 目录 树 导 航 的 类 或 方法 ; 

口 设 办 法 检测 和 处 理 符 号 链接 ; ” 

口 用 简单 操作 读 不 出 文件 的 属性 ( 比如 可 读 、 可 写 或 可 执行 )。 

用 Java 7 的 NIO.2 API 可 以 完成 这 个 不 可 能 的 编程 任务 ， 它 支持 目录 树 的 直接 导航 (Files. 
walkFileTree() ，2.3.1 节 )、 符 号 链接 (Files.isSymbolicLink()， 代 码 清单 2-4 )， 能 用 一 
行 代码 读 取 文件 属性 (Files.readattributes()，2.4.3 节 )。 

除 此 之 外 , 老板 还 要 求 你 在 读 取 这 些 properties 文 件 时 不 能 打 断 主 程序 的 处 理 流 程 。 可 最 小 的 
properties 文 件 也 有 1MB， 读 取 这 些 文件 很 可 能 打 靳 程序 的 主流 程 ! 面 对 这 一 要 求 ， 在 Java 5/6 的 
时 代 ， 你 很 可 能 会 用 java .util .concurrent 包 中 的 类 创建 线程 池 和 工作 线程 队列 ， 再 用 单独 


山 符号 链接 是 一 种 特殊 类 型 的 文件 ， 指 向 文件 系统 中 的 另外 一 个 文件 或 位 置 一 一 你 可 以 把 它 理解 为 快捷 方式 。 
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的 后 台 线 程 读 取 文件 。 我 们 在 第 4 章 将 会 讨论 到 ， 现 在 Java 中 的 并 发 仍然 相当 困难 ， 并 是 非常 容 
易 出 错 。 人 借助 Java 7 和 NIO.2 API， 你 可 以 用 新 的 AsvnchronousFilechannel (2.5 人 )， 不 用 指 
定 工作 线程 或 队列 就 可 以 在 后 台 读 取 大 型 文件 。 啉 ! 

这 个 新 API 非 常 有 用 ， 尽 管 它 不 能 帮 你 冲 咖啡 、 但 它 的 发 展 趋势 可 在 那儿 搜 着 呢 。 

第 一 个 趋势 是 对 其 他 数据 存储 方法 的 探索 , 特别 是 在 非 关 系 或 大 数据 集 领域 。 你 可 能 很 快 就 
会 遇 到 读 写 大 文件 (比如 微 博 上 的 大 型 报告 文件 ) 的 问题 。NIO.2 可 以 帮助 你 用 一 种 异步 、 有 效 
的 方式 读 写 大 文件 ， 还 能 利用 底层 操作 系统 的 特性 。 

第 二 个 趋势 是 多 核 CPU 的 发 展 ， 使 得 真正 并 发 且 和 更 快 的 MO 成 为 可 能 。 并 发 是 个 难以 掌握 的 
领域 ,但 NIO.2 会 助 你 一 臂 之 力 , 它 为 多 线程 文件 和 套 接 字 访问 的 应 用 提供 了 一 个 简单 的 抽象 层 。 
即使 你 不 直接 使 用 这 些 特 性 ， 它 们 也 会 对 你 的 编程 生 淮 产 生 极 大 影响 ， 因 为 IDE、 应 用 服务 辫 和 
各 种 流行 的 框架 会 大 量 应 用 这 些 特 性 。 

这 些 只 是 NIO.2 会 对 你 有 哪些 帮助 的 例子 。 如果 NIO.2 可 以 解决 你 眼下 面临 的 一 些 问 题 , 本 章 
的 内 容 就 是 为 你 准备 的 ! 否则 ， 你 可 以 在 接 到 Java LO 任务 时 再 回来 。 

本 章 你 会 体验 到 Java 7 新 LO 的 能 力 ， 以 便 你 能 够 开始 编写 基于 NIO.2 的 代码 ， 并 有 信心 探索 
新 的 API。 除 此 之 外 ， 这 些 API 还 使 用 了 一 些 第 1 章 提 到 的 特性 ， 这 证 明 Java 7 确实 会 使 用 自己 的 


提示 “将 try-with-resoureces 和 NIO,2 中 的 新 API 结 合 起 来 可 以 写 出 非常 安全 的 IO 程序 ， 这 在 Java 
中 还 是 破天荒 的 第 一 次 ! 


我 们 觉得 你 很 可 能 会 用 到 新 的 文件 O 能 力 ， 所 以 本 章 会 非常 详细 地 介绍 。 你 需要 从 了 解 新 
的 文件 系统 抽象 屋 开始 ， 即 先 了 解 Path 和 它 的 辅助 类 。 在 path 之 上 ， 你 会 接触 到 常用 的 文件 系 
统 操作 ， 比 如 复制 和 移动 文件 。 

我 们 还 会 向 你 介绍 异步 1O， 给 你 看 一 个 文件 系统 的 例子 。 最 后 我 们 会 讨论 套 接 字 和 通道 功 
能 的 融合 ， 以 及 这 对 于 网 络 应 用 开发 人 员 意味 着 什么 。 但 我 们 先 来 看 一 下 NIO.2 的 由 来 。 


2.1 Java I/O 简 史 


要 品 出 NIO.2 API 设 计 的 真正 味道 ， 并 深刻 理解 它们 的 用 法 ， 应 该 先 弄 清 Java LO 的 历史 。 但 
我 们 非常 理解 你 想 要 接触 代码 的 渴望 。 别 着 急 ， 你 的 这 种 渴望 可 以 在 2.2 节 得 到 满足 。 

如 果 你 发 现 某 些 API 的 用 法 非常 简单 甚至 有 点 古怪 ， 本 节 会 帮助 你 从 API 设 计 者 的 角度 看 待 
NIO.2。 这 是 Java 1/O0 的 第 三 个 主要 版 本 ， 所 以 让 我 们 回顾 一 下 Java 对 LO 支持 的 发 展 历史 ， 看 看 
NIO.2 是 怎么 产生 的 。 

Java 之 所 以 能 够 广泛 流传 ， 其 强大 、 丰 富 、 简 明 的 类 库 功 不 可 没 ， 编程 时 要 解决 的 大 多 数 问 


DD 第 4 章 深 入 探讨 了 并 发 计算 可 能 给 你 的 编程 生 淮 带 来 的 微妙 复杂 性 。 
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题 几乎 都 可 以 在 其 中 找到 支持 。 但 经 验 丰 富 的 Java 开 发 人 员 都 知道 ， 在 老 版 本 的 Java 中 ， 有 些 地 
方 不 是 那么 给 力 。 曾 经 让 他 们 最 崩溃 的 就 是 Java 的 输 和 人 /输出 【LO ) APL。 


2.1.1 Java 1.0 到 1.3 


在 Java 的 早期 版 本 ( 1.0-1.3 ) 中 设 有 完整 的 IO 支持 。 也 就 是 说 开发 人 员 在 开发 需要 IO 支持 
的 应 用 时 ， 要 面临 如 下 问题 。 

口 没有 数据 缓冲 区 或 通道 的 概念 ， 开 发 人 员 要 编程 处 理 很 多 底层 细节 。 

UO 操 作 会 被 阻塞 ， 扩 展 能 力 受 限 。 

口 所 支持 的 字符 集 编码 有 限 ， 需 要 进行 很 多 手工 编码 工作 来 支持 特定 类 型 的 硬件 。 

口 不 支持 正则 表达 式 ， 数 据 处 理 困难 。 

基本 上 , 早期 版 本 的 Java 人 缺乏 对 非 阻 塞 VO 的 支持 。 开 发 人 员 万 般 无 奈 , 只 好 自己 实现 可 伸缩 
的 LO 解决 方案 。 在 Java 1.4 发 布 之 前 ，Java 一 直 没 能 在 服务 器 端 开 发 领域 得 到 重用 ， 我 们 认为 主 
要 原因 就 是 缺乏 对 非 阻塞 JO 的 支持 。 


2.1.2 在 Java 1.4 中 引入 的 NIO 


为 了 解决 这 些 问 题 ，Java 开 始 实现 对 非 阻塞 UO 的 支持 ， 与 其 他 LO 特性 一 起 ， 帮 助 开 发 人 员 
交付 更 快 、 更 可 靠 的 JO 解 决 方案 。 其 中 有 两 次 主要 改进 ， 

D 在 Java 1.4 中 引 人 非 阻塞 LO; 

口 在 Java 7 中 对 非 阻塞 VO 进 行 修 改 。 

2002 年 发 布 Java 1.4 时 ,， 非 阻塞 LO (NIO ) 以 JSR-S1 的 身份 加 入 到 Java 语 言 中 。 下 面 这 些 特性 
就 是 那 时 增加 的 。 自 此 之 后 ，Javai 语 言 终于 得 到 了 服务 器 端 开 发 人 员 的 青睐 : 

口 为 UO 操作 抽象 出 缓冲 区 和 通道 层 ; 

口 字符 集 的 编码 和 解码 能 力 ; 

口 提供 了 能 够 将 文件 映射 为 内 存 数据 的 接口 ; 

口 实现 非 阻塞 IO 的 能 力 ; 

口 基于 流行 的 Perl 实 现 的 正则 表达 式 类 库 。 
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。 毋庸 置疑 ，Perl 编 程 语言 是 正则 表达 式 处 理 之 王 。 实 际 上 ， 它 的 设计 和 实现 相当 出 多， 其 
至 很 多 编程 语言 ( 包括 Java ) 竟 丰 模仿 Per 的 语法 和 语义 。 如 果 体 对 Perl 语 言 感 兴趣 ， 可 以 访问 
http://www.perlorg/ 一 探究 竟 。 


NIO 无 疑 使 Java 回 前 迈 出 了 一 大 步 ， 但 LO 编程 对 Java 开 发 人 员 来 说 仍然 是 个 挑战 。 特 别 是 对 
于 文件 系统 中 的 文件 和 目录 处 理 而 言 ， 支 持 力度 还 是 不 够 。 那 时 的 java.io.File 类 有 些 比 较 烦 
人 的 局 限 性 。 
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口 在 不 同 的 平台 中 对 文件 名 的 处 理 不 一 致 。” 

口 没有 统一 的 文件 属性 模型 。( 比如 读 写 访问 模型 ) 
口 遍历 目录 困难 。 

口 不 能 使 用 平台 /操作 系统 的 特性 。* 

口 不 支持 文件 系统 的 非 阻塞 操作 。” 


2.1.3 下 一 代 IO-NIO.2 


为 了 突破 这 些 局 限 性 ， 同 时 也 为 了 支持 现代 硬件 和 软件 1/O 的 新 范例 ， 由 阿兰 * 波 特 曼 主 导 的 
JSR-203 应 运 而 生 。JSR-203 最 终 变 成 了 我 们 在 Java 7 中 见 到 的 NIO.2 API。 它 有 三 个 主要 目标 ， 
JSR-203 规 范 中 的 第 2.1 节 对 它们 进行 了 详细 的 介绍 ( http://jcp.org/en/jsr/detail?id=203 ) 

(1) 一 个 能 批量 获取 文件 属性 的 文件 系统 接口 ,去掉 和 特定 文件 系统 相关 的 AP1, 还 有 一 个 用 
于 引入 标准 文件 系统 实现 的 服务 提供 者 接口 。 

(2) 提供 一 个 套 接 字 和 文件 都 能 够 进行 异步 ( 与 轮 询 、 非 阻塞 相对 ) WO 操作 的 API。 

(3) 完成 JSR-5S1 中 定义 的 套 接 字 一 一 通道 功能 ， 包 括 御 外 对 绑 定 、 选 项 配置 和 多 播 数 据 报 的 
支持 。 

接 下 来 让 我 们 从 新 文件 系统 的 基础 Path 和 它 的 朋友 们 开始 吧 ! 


2.2 文件 IO 的 基石 : Path 


在 N10.2 的 文件 W/O 中 ，Path 是 必须 掌握 的 关键 类 之 一 。Path 通 常 代表 文件 系统 中 的 位 置 ， 
比如 C:\workspaceijava7developer ( Windows 文 件 系统 中 的 目录 ) 或 /usr/bin/zip ( *nix 文 件 系统 中 zip 
程序 的 位 置 ) 如 果 你 能 理解 如 何 创建 和 处 理 路 径 ， 就 能 浏览 任何 类 型 的 文件 系统 ， 包 括 zaip 归 档 
文件 系统 。 

我 们 通过 图 2-1 ( 基于 本 书 中 的 源码 布局 ) 来 复习 一 下 文件 系统 中 的 几 个 概念 。 

口 目录 树 

口 根 路 径 

口 绝对 路 径 

口 相对 路 径 

之 所 以 要 讨论 绝对 路 径 和 相对 路 径 ， 是 因为 我 们 需要 考虑 程序 运行 的 位 置 。 比 如 说 ， 有 个 程 
序 可 能 在 /java7developer/src/test 目 录 下 运行 ， 代 人 码 要 读 取 位 于 /java7developer/src/main 目 录 下 的 文 
件 名 。 为 了 进入 /java7developer/src/main 目 录 ， 可 以 用 相对 路 径 ../main，。 

但 如 果 程 序 运行 在 /java7developer'src/testjava/conyjava7developer， 用 相对 路 径 .Jmain 则 无 法 


DD 一些 批评 者 会 说 Java 没 有 兑现 “一 次 编写 ,处 处 运行 ”的 承诺 
到 人 们 通常 希望 能 够 使 用 LinuwUNIX 中 的 符号 链接 机 制 。 
0 Java 1.4 的 确 支 持 网 络 套 接 宇 的 非 阻 塞 操 作 。 
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进入 目标 目录 ， 而 会 进 到 并 不 存在 的 目录 /java7Tdeveloper/src/testijava/com/main 中 。 所 以 必须 使 用 
绝对 路 径 ， 比 如 /java7developer/src/main。 

反之 亦 然 ， 你 的 程序 可 能 一 直 运 行 在 同一 位 置 ， 比 如 图 2-1 中 的 target 目 录 。 但 目录 树 的 根 目 
录 可 能 会 变 ， 比 如 从 /java7developer 变 成 了 D:workspaceNj7d。 这 时 就 不 能 依靠 绝对 路 径 ， 而 要 用 
相对 路 径 转 到 你 想到 达 的 位 置 。 


人 文件 系统 的 根 目录 > 


1 
— java7developer/ 

|-- llby 

[= pom.xml 

| sample_posix build.properties 

| sampie_ windows_build.properties 
从 文件 系统 的 根 目 录 到 java 目 录 的 绝对 路 径 是 
/javardeveloper/sre/main/java/ 


: Using 2 1.java 


-- -chaplar5i 

-Listing 5 1.java 
Listing 5 2java 

一 一 resources/ 

一 scalal 


i 
日 
中 
下 
下 
ee 
证 


det 要 从 test 目 录 到 main 目 录 中 去 ， 可 以 
a 。 经 过 相对 路 径 ../main 一 一 也 就 是 说 
~ javal 回 到 上 一 级 的 sre (../ 中 ， 然 后 再 
i 于 入 main 
一 人 avardewvelopenm 
一 chapter1/ 
一 scalal 
== CDOmU 
— java7developer/ 
=— Chapterg/ 
‘Listing 9 1.scala 
一 Listing 9 2.scala 


一 target/ 
图 2-1 说明 根 目录 、 绝 对 路 径 和 相对 路 径 概念 的 目录 树 


NIO.2 中 的 Path 是 一 个 抽象 构造 。 你 所 创建 和 处 理 的 Path 可 以 不 马上 绑 定 到 对 应 的 物理 位 置 
上 上。 尽管 看 起 来 奇怪 ,但 有 时 确实 需要 如 此 。 比 如 说 ， 你 想 创建 一 个 Path 来 表示 即将 创建 的 新 


只 自 在 读书 @3 
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文件 。 在 调用 Files .createFile(Path targqet)“ 之前， 这 个 文件 是 不 存在 的 。 如 果 在 Path 
所 对 应 的 文件 创建 之 前 ， 你 试图 读 取 这 个 文件 中 的 内 容 ， 就 会 导致 TOException。 如 果 你 指定 
了 一 个 并 不 存在 的 Path 并 试图 用 Files.readAllBytes (Path) 之 类 的 方法 读 取 ， 纺 果 是 一 梓 
的 。 简 言 之 ，JVM 只 会 把 Path 绑 定 到 运行 时 的 物理 位 置 上 。 


警告 ”在 编写 与 具体 文件 系统 相关 的 代码 时 要 小 心 。 创 建 Path 为 C'vworkspaceMjava7developer， 
然后 试图 读 取 它 的 程序 ， 这 只 能 在 有 C:\workspace\java7developer 位 置 的 计算 机 上 才能 工 
作 。 一定 要 确保 程序 逻辑 和 异常 处 理 考虑 到 了 各 种 情况 ， 包 揪 可 能 在 不 同 的 文件 系统 上 
运行 ， 或 文件 系统 的 结构 可 能 被 改动 。 本 书 的 作者 之 一 过 去 就 犯 过 这 个 错误 ， 结 果 把 计 
算 机 系 的 一 组 硬盘 全 搞 坏 了 ! 


还 是 得 再 重复 一 遍 ，NIO.2 把 位 置 ( 由 Path 表 示 ) 的 概念 和 物理 文件 系统 的 处 理 ( 比如 复制 
一 个 文件 ) 分 得 很 清楚 ， 物 理 文件 系统 的 处 理 通常 是 由 Files 辅 助 类 实现 的 。 
有 关 Path 类 以 及 将 在 本 节 讨 论 的 其 他 类 的 进一步 细节 请 见 表 2-1。 
表 2-1 学 习 文 件 I/O 的 关键 基础 类 
到 说 明 
Path Path 业 中 的 方法 可 以 用 素 获 取 路 径 信 息 ， 访 问 该 路 空中 的 各 元 素 ， 将 路 径 转 换 为 其 他 形式 ， 
或 提取 路 径 中 的 一 部 分 。 有 的 方法 还 可 以 匹配 路 径 字 串 以 及 移 除 路 径 中 的 元 余 项 
patha 工具 类 ， 提 供 返 回 一 个 路 径 的 辅助 方法 ， 比 加 get (String first, String..， more) 和 
get (URI uri) 
FileSystem 与 文件 系统 交互 的 类 ， 无 论 是 默认 的 文件 系统 ， 还 是 通过 其 统一 资产 标识 (URI) 获取 的 可 
选 文件 系统 
FileSystems 工具 类 ， 提 供 各 种 方法 ， 比 如 其 中 用 于 返回 点 认 立 件 系 统 的 FileSystems .getDefault 1() 


记 住 ，Path 不 一 定 代 表 真实 的 文件 或 目录 。 你 可 以 随心 所 欲 地 操作 Path， 用 Files 中 的 功 
能 来 检查 文件 是 否 存 在 ， 并 对 它 进行 处 理 。 


提示 。 Path 并 不 仅 限于 传统 的 文件 系统 ， 它 也 能 表示 zip 或 iar 这 样 的 文件 系统 。 


我 们 来 完成 几 个 简单 的 任务 ， 探 索 一 下 Path 类 : 
口 创建 一 个 Path; 

口 获取 Path 的 相关 信息 ; 

口 移 除 Fath 中 的 元 余 项 ; 


了 你 一 会 儿 就 能 在 2.4 节 见 到 这 个 方法 | 
回 我 们 不 会 告诉 你 是 谁 ， 不 过 欢迎 你 把 他 查 出 来 ! 
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口 转换 一 个 Path: 
口 合并 两 个 Path， 在 两 个 Path 中 间 创 建 一 个 Path， 并 对 这 两 个 Path 进行 比较 。 
我 们 先 从 创建 表示 文件 系统 中 某 个 位 置 的 Path 开 始 。 


2.2.1 创建 一 个 Path 


创建 Path 没 什么 难 的 。 调 用 paths .get (string first,String...more) 是 最 快 
的 做 法 ， 其 中 第 二 个 变量 一 般 用 不 到 ， 它 仅仅 用 来 把 额外 的 字符 串 合 并 起 来 形成 Path 字符 串 。 


提示 在 NIO.2 的 API 中 ，Path 或 Paths 中 的 各 种 方法 抛 出 的 受 检 异常 只 有 IOException。 我 们 
认为 这 虽然 简单 ， 但 有 时 却 会 掩藏 潜在 的 问题 ， 而 且 如 果 你 想 处 理 IOException 的 某 个 
显 式 子 类 ， 则 需要 额外 编写 异常 处 理 代码 。 


我 们 用 Paths .get(String first) 方 法 为 /usrbin/ 目 录 下 的 文件 压缩 工具 zip 创 建 一 个 绝对 
Path, 

Path listing = Paths.get ("/usr/bin/zip"); 

调用 Paths .get (" /usr/bin/zip") 的 效果 和 调用 下 面 这 个 长 一 点 的 代码 效果 一 样 : 


Path listing = FileSystems.getDefault() .getPath("/usr/bin/zip"); 


提示 “创建 Path 时 可 以 用 相对 路 径 。 比 如 ， 运 行 在 /opt 目 录 下 的 程序 可 以 用 .Jusrbin/zip 创 建 一 
个 指向 /usrbin/zip 的 Path。 其 含义 是 指 进 入 /opt 的 上 一 层 目 录 ( 即 /) 然后 进入 /usrbin/zip。 
通过 调用 toabsolutepPath1) 方 法 ,很 容 友 把 这 个 相对 路 径 转 摘 成 绝对 路 径 ， 如 ， 
listing.toAbsolutePath!()。 


你 可 以 从 Path 中 获取 信息 ， 比 如 其 父 目录 、 文 件 名 ( 如果 有 的 话 ) 等 。 


2.2.2 从 Path 中 获取 信息 


Path 类 中 有 一 组 方法 可 以 返回 你 正在 处 理 的 路 径 的 相关 信息 。 代 码 清 单 2-1 为 /uswbin 目 录 下 
的 zip 工 具 创 建 一 个 Path， 并 输出 相关 信息 ， 包 括 它 的 根 目 录 和 父 目 录 。 如 果 你 的 操作 系统 中 zip 
也 放 在 /usr/bin 目 录 下 ， 应 该 能 见 到 下 面 这 种 输出 信息 。 

File Name [zip] 

Number of Name Elementes in the Path [3] 

Parent Path [/usr/bin] 


Root of Path [/] 
Subpath from Root, 2 elements deep [usr/bin] 


你 在 计算 机 上 看 到 的 结果 和 你 所 用 的 操作 系统 及 程序 运行 的 位 置 有 关 。 
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代码 清单 2-1 从 Path 中 获取 信息 


import java.nio.file.Path; 创建 绝对 路 径 


import java.nio.file.Pathe; 
public clasgs Lieting 2 1 | 


public static void main(Sstring[] args) | 
Path listing = Paths,.get ("/usr/bin/zip"),; 
system.out .println("File Name |[" + 
listing.getFileName() +* "]"™); 
System.out .println("Number of Name Elements 
in the Path [™ + 


listing.getNameCount() + "]"); 
Syatem.out .println("Parent Path [" * 
listing.getParent () + "]"™); 
Syatem.out .println("Root of Path [" + i | 
和 2 | pes () 本 "]")s ’ gi 
System.out .println("Subpath from Root, i 
2 elements deep |[" + 
listing.subpath(0, 2) + "]"); 
] 
! 
在 创建 了 /usr/bin/zip 的 Path 之 后 , 你 可 以 看 一 下 组 成 Path 的 元 素 个 数 , 在 此 例 中 就 是 目录 的 
数量 力 。 相 对 其 父 目录 和 根 目 录 ，Path 的 位 置 会 更 有 用 。 也 可 以 通过 指定 起 娘 和 终止 的 索引 来 
挑 出 子路 径 。 在 本 例 中 是 从 Path 的 根 (0) 到 其 第 二 个 元 素 (2 ) 之 间 的 子路 径 @@。 
如 果 这 是 你 初次 接触 NIO.2 文 件 API, 这 些 方法 对 你 来 说 非常 重要 , 因为 你 可 以 借助 它们 查看 
路 径 处 理 的 结果 。 


2.2.3 ” 移 除 元 余 项 


在 编写 工具 软件 ， 比 如 属性 文件 分 析 几 时， 需要 处 理 的 Path 中 可 能 会 有 一 个 或 两 个 点 : 

口 . 表示 当前 目录 ; 

.. 表示 父 目 录 。 

假设 你 的 程序 在 /java7developer/src/main/java/com/java7developer/chapter2/ 目 录 下 运行 ( 见 图 
2-1 )。 你 所 在 的 目录 和 Listing_2_1 .java 一 样 , 如 果 传 给 你 的 Path 是 . /Listing_2.1.Jjava， 
其 中 的 . /部 分 ， 即 程序 正在 运行 的 目录 ， 实 际 上 并 没什么 用 。 在 这 里 用 短 一 点 儿 的 Path 一 一 
Listing 2 1.java 就 够 了 。 

Path 中 可 能 还 有 其 他 元 余 项 , 比如 符号 链接 ( 见 2.4.3 节 )。 假设 在 *nix 系 统 中 /srlogs 目 录 下 ， 
你 想 寻 找 日 志文 件 logl.txt， 但 其 实 /usr/logs 只 是 一 个 指 问 /application/logs 的 符号 链接 ， 那 里 才 是 
存放 日 志文 件 的 真正 位 置 。 要 得 到 这 个 位 置 ， 就 需要 去 掉 元 杂 的 符号 信息 。 

所 有 这 些 元 余 项 都 会 导致 path 指 向 的 不 是 你 认为 它 应 该 指向 的 位 置 。 

在 Java 7 中 , 有 了 两 个 辅助 方法 可 以 用 来 弄 清 path 的 真实 位 置 。 首 先 可 以 用 ncrmalizel() 方 法 
去 掉 Path 中 的 元 余 信 息 。 下 面 的 代码 会 返回 Listing 2 1.java 的 Path， 去 掉 了 表明 它 在 当前 目录 
( ./ 部 分 ) 中 的 元 余 符号 。 
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Path normalizedPath = Paths.get("./Listing 2 1.]ava") .normalize (); 

此 外 ，toRealPath() 方 法 也 很 有 效 ， 它 融合 了 toAbsolutePath() 和 normalize() 两 个 
方法 的 功能 ， 还 能 检测 并 跟随 符号 连接 。 

还 是 回 到 *nix 系 统 中 的 日 志文 件 那个 例子 ， 在 /usrlogs 目 录 下 有 个 日 志文 件 logl.txt， 而 这 个 
目录 实际 上 是 指 问 /application/logs 的 符号 链接 。 通 过 调用 toRealPath() ， 你 能 得 到 表示 
/application/logs/log]1.txt 的 真正 Path。 

Path realPath = Pathas.,.get ("/usr/logs/logl.txt") .toRealPathl):; 


我 们 要 讨论 的 最 后 一 个 Path API 特 性 是 比较 多 个 Path， 并 找 出 它们 之 间 的 Path。 


2.2.4 转换 Path 


转换 路 径 最 多 的 是 工具 软件 。 比 如 你 可 能 需要 比较 文件 之 间 的 关系 , 以便 了 解 源码 目录 树 的 
结构 是 否 符 合 编码 规范 。 或 者 在 shell 脚 本 中 执行 程序 时 可 能 会 传人 一 些 Path 参 数 , 并 需要 把 这 些 
参数 变 成 有 意义 的 Path。 在 NIO.2 里 可 以 很 容易 地 合并 Path， 在 两 个 Path 中 再 创建 Path 或 对 
Path 进 行 比 较 。 

下 面 的 代码 将 两 个 Path 合 并 ， 通 过 调用 resolve 方 法 ， 将 uat 和 conf/application.properties 合 
并 成 表示 /uat/conf/application.properties 的 完整 Path。 


path prefix = Paths.get ("/uat/"):; 
Path completePath = prefix.resolve ("conf /application .properties"):; 


要 取得 两 个 Path 之 间 的 路 径 ， 可 以 用 relativize (Path) 方 法 。 下 面 的 代码 计算 了 从 日 志 
目录 到 配置 目录 之 加 的 路 径 。 


String logging = args (0]; 

String configuration = args[1]; 

Path logDir = Paths.get (logging):; 

Path confDir = Pathas.get Iconfiguration); 

Path pathToConfDir = logDir.relativize lconfDir); 


如 你 所 愿 ， 你 可 以 用 startsWith(Path prefix) 、equals (Path path) 等 值 比较 或 
endsWith(Path suffix) 玉 对 蹄 行进 行 比 较 。 

现在 使 用 Path 类 对 你 来 说 应 该 设 有 问题 了 ， 但 Java 7 之 前 版 本 的 那些 代码 怎么 办 呢 ?” 负责 
NIO.2 的 团队 考虑 到 了 向 后 兼容 性 ， 所 以 加 了 两 个 新 的 API 特 性 来 保证 基于 Path 的 新 LO 和 老 版 本 
代码 之 间 的 互 操作 性 。 


2.2.5 NIO.2 Path 和 Java 已 有 的 File 类 


新 API 中 的 类 可 以 完全 替代 过 去 基于 java.io.File 的 API。 但 你 不 可 避免 地 要 和 大 量 遗 留 下 
来 的 LO 代码 交互 。Java 7 提供 了 两 个 新 方法 : 

D java .io.File 类 中 新 增 了 toPath() 方 法 , 它 可 以 马上 把 已 有 的 File 转 化 为 新 的 Path。 

口 Path 类 中 有 个 toFile() 方 法 ， 它 可 以 马上 把 已 有 的 Path 转 化 为 File。 
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下 面 的 代码 演示 了 这 一 功能 。 


File file = new File("../Listing 2 1.java").; 
Path listing = file.toPathl(); 
listing.toAbsolutePpathl(), 

file = listing.toFile!(),; 


现在 ， 我 们 完成 了 对 Path 类 的 探索 。 接 着 要 研究 一 下 Java 7 对 目录 处 理 芯 


2.3 ”处理 目 录 和 目录 树 


在 读 2.2 节 关于 路 径 的 内 容 时 ， 你 可 能 已 经 猜 到 目录 不 过 是 带 有 特别 属性 的 Pacth。 所 有 历 目录 
的 能 力 是 Java 7 引 人 注 目的 新 特性 。 新 加 入 的 java .nio.file.DirectoryStream<T> 接 口 和 它 
的 实现 类 提供 了 很 多 功能 : 

口 循环 遍历 目录 中 的 子 项 ， 比 如 查找 目录 中 的 文件 ; 

口 用 glob 表 达 式 ”( 比如 *Foobar* ) 进行 目录 子 项 匹配 和 基于 ) 

text /yml 文 件 ); 

口 用 walkFileTree 方 法 实现 好 归 移动 、 复 制 和 删除 操作 。 

本 下 主要 讨论 两 个 常见 用 例 : 在 一 个 目录 中 查找 文件 以 及 在 目录 树 中 执行 相同 的 任务 。 我 们 
先 从 最 简单 的 开始 : 在 一 个 目录 中 查找 任意 文件 。 


2.3.1 在 目录 中 查找 文件 

们 先 讨论 一 个 简单 的 例子 , 用 模式 匹配 过 滤 出 java7developer 项 目 中 所 有 的 .properties 文 件 。 
请 看 下 面 的 代码 ; 
代码 清单 2-2” 列 出 目录 下 的 properties 文 件 


Path dir = Paths.get ("CcC:\\workspace\\java7developer"); 4 天 设 定 起 始 路 径 


ME 的 内 容 检测 ( 比如 


trvylDirectorySstream< Path>s stream 
= Files.newDirectoryStream(ldir, "*.properties")) | 中 一 
for (Path entry: stream) | | 声明 过 淖 流 


Syasetem.out .printlnlentry.getFileName()); 


找 出 所 有 .properties 


文件 并 输出 


catcecn (IOQEXCception e€) 


| 


System,out .printlnle.getMessgage())}); 


最 前 面 是 我 们 已 经 熟悉 的 Paths .get (string) 调 用人。 紧 随 其 后 的 是 关键 方法 Direct- 


山 gl1ob 通 常 指 有 限 的 模式 匹配 ， 源 自 早 期 Unix 中 的 glob1) 库 函数 ,该 函数 用 于 查找 文件 系统 中 指定 模式 的 路 徐 ， 
虽然 所 用 语法 和 正则 表达 式 类 似 ， 但 没有 正则 表达 式 那么 强大 的 表达 力 ， 详 见 附录 B。 一 一 译 者 注 
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oryStream(Path directory, String patternMatch) 各 ， 它 返回 一 个 经 过 过 滤 的 
DirectoryStream， 上 其 中 包含 以 .properties 结 尾 的 文件 。 最 后 输出 每 个 子 项 咎 . 

过 滤 流 中 用 到 的 模式 匹配 称 为 glob 模 式 匹 配 , 它 和 Perl 正 则 表达 式 类 似 , 但 稍 有 不 同 。 附 录 B 
中 有 如 何 使 用 glob 模 式 匹 配 的 详细 说 明 。 

代码 清单 2-2 展 示 了 新 API 处 理 单个 目录 的 能 力 , 但 如 果 需 要 递归 过 滤 多 个 目录 时 该 怎么 办 ? 


2.3.2 ”遍历 目录 树 


Java 7 支持 整个 目录 树 的 遍历 。 也 就 是 说 你 可 以 很 容易 地 搜寻 目录 树 中 的 文件 ， 在 于 目录 中 
查找 , 并 对 它们 执行 操作 。 比 如 你 可 能 想 在 做 开发 的 机 静 上 弄 个 工具 类 来 删除 目录 /optUworkspacey 
java 下 的 所 有 .class 文 件 ， 完 成 构建 前 的 清除 工作 。 

遍历 目录 树 是 Java 7 的 新 特性 ， 要 想 正确 使 用 ， 你 得 掌握 一 些 接口 及 其 实现 的 细节 。 其 中 的 
关键 方法 是 : 

Files.walkFileTree (Path startingDir, FileVvisitor<? Super Path> visitor); 

提供 startingDir 非 常 简单 ， 但 给 出 Filevisitor 接 口 的 实现 类 就 比较 麻烦 了 ( 参数 
FileVisitor<? superPath> visitor 看 上 去 就 不 是 着 苦 儿 )， 因 为 最 少 得 实现 下 面 5 个 方法 
(TT 一般 就 是 Path ): 

上 FileVvisitResult prevVvisitDirectory(T dir) 

FileVisitResult preVisitDirectoryFailed(T dir, IOException exc) 

LL FileVisitResult visitFilel(lT file, BasicFileAttributes attrs) 

DD FilevVvisitResult visitFileFailed(T file, IOException exc) 

0D FileVvisitResult postVisitDirectory(T dir, IOException exc) 

看 起 来 挺 吓 人 的 吧 ? 好 在 Java 7 API 的 设计 者 们 已 经 提供 了 一 个 默认 实现 类 ，simpleFile- 
Visitor<T>o 

我 们 要 扩展 并 修改 代码 清单 2-2。 下 面 的 代码 会 列 出 C:\workspace\java7developer\src 目 录 下 及 
其 子 目录 内 的 所 有 .java 文 件 , 这 有 段 代码 展示 了 Files .walkFileTree 方 法 对 默认 实现 类 simple- 
FileVisitor 的 用 法 ， 用 一 个 特定 的 visitFile 方 法 实现 来 改进 它 。 


代码 清单 2-3 ” 列 出 于 目录 下 的 所 有 java 源 码 文件 


PUblic class Find 


| 

public static void mainf(Strino[] args) throws IOExcept ion 二 

{ a 3 a 设置 起 始 目 录 
Path startingDir = 

Paths.get ["C:\\workspace\\Java7Tdeveloper\\src"):; 可 
Files.walkrileTree (startingDir, | 
new FindJavaVvisitor|)); 辣 调用 walLkFileTree 
| 


private static class FindJavaVisitor 


1 入 
三 


28 第 2 章 新 LO 


extends SimpleFileVvisitor<Path> 扩展 gimplerile- 
Visitor <Path> 


eeOverride 
Public FlileVisitResult 


visitFilelPath file, BasicFileAttributes attra) 
{ 重 写 wisitFile 
if (file.tostring() .endewith{".java")) i 方法 


System.out .printlnlfile.getFileName!()),; 


return FileVisitReasult .CONTINUE:; 
} 


| 
} 


刺 个 过 程 从 调用 Files .walkFileTree 方 法 开始 @@， 这 里 的 关键 是 FindJavaVisitor， 该 
类 扩展 了 Filevisitor 的 默认 实现 类 simpleFilevisitor 合 。 你 相让 simpleFilevVisitor 来 
完成 大 部 分 工作 ， 比 如 遍历 目录 。 可 实际 上 你 唯一 要 做 的 就 是 重 写 visitFile(Path, 
BasicFileAttributes)" 方 法 合 , 在 这 个 方法 中 你 也 只 需要 写 些 代 码 来 判断 文件 名 是 否 以 java 
结尾 ， 如 果 确 实 是 ， 就 在 stdout 中 输出 。 

其 他 用 例 包括 递归 移动 、 复 制 、 删 除 或 者 修改 文件 。 在 大 多 数 应 用 场景 中 ， 你 只 需要 扩展 


simpleFileVisitor。 但 如 果 你 想 实 现 自己 的 FileVisitor，API 也 很 灵活 。 


注意 为 了 确保 递归 等 操作 的 安全 性 ，walkFileTree 方 法 不 会 自动 跟随 符号 链接 。 如 果 你 确 
实 需要 跟随 符号 链接 ， 就 需要 检查 那个 属性 ( 如 2.4.3 节 所 述 ) 的 本 人。 


现在 你 对 路 径 和 目录 树 已 经 熟悉 了 , 该 从 处 理 位 置 进 人 真正 的 文件 系统 操作 上 了 , 接 下 来 我 
们 会 同 你 介绍 新 的 Files 类 及 它 的 朋友 们 。 
2.4 NIO.2 的 文件 系统 |/O 


对 于 文件 系统 的 操作 任务 ， 比 如 移动 文件 、 修 改 文 件 属性 ， 以 及 处 理 文 件 内 容 等 ， 在 NIO.2 
中 都 有 所 改善 。 对 这 些 操作 的 支持 主要 是 由 Files 类 提供 的 。 
表 2-2 中 有 关于 Files 类 的 详细 人 介绍， 此 外 本 节 还 会 介绍 另外 一 个 也 很 重要 的 类 ， 


WatchService, 


表 2-2 文件 处 理 的 基础 类 
类 : 说 上 明 


Piloe 让 你 轻松 复制 、 移 动 ， 删 除 或 处 理 文件 的 工具 类 ， 有 你 需要 的 所 有 方法 
WatchService 用 来 监视 文件 或 目录 的 核心 类 ， 不 管 它们 有 没有 变化 


D2.4 节 会 介绍 BasicFileAttributes， 所 以 暂时 不 用 管 它 。 
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在 本 节 中 ， 你 将 学 会 如 何在 文件 和 文件 系统 上 执行 下 面 这 些 任 务 : 

口 创建 和 删除 文件 ; 

口 移动 、 复 制 、 重 命名 和 删除 文件 ; 

口 文件 属性 的 读 写 ; 

口 文件 内 容 的 该 取 和 写 和 人; 

口 处 理 符号 链接 ; 

口 用 watchservice 发 出 文件 修改 通知 ; 

口 使 用 seekableBvytechannel 个 可 以 指定 位 置 及 大 小 的 增强 型 字 节 通道 。 

这 看 起 来 可 能 挺 已 怖 的 , 但 由 于 设计 巧妙 ，API 提 供 了 很 多 辅助 方法 ,把 抽象 层 隐 藏 了 起 来 ， 
让 你 可 以 轻松 快捷 地 处 理 文件 系统 。 


警告 NIO.2 API 对 原子 操作 的 支持 有 很 大 改进 ， 但 涉及 文件 系统 处 理 时 ， 仍 然 主要 依靠 代码 来 
提供 保护 。 即 使 是 执行 了 一 夺 的 操作 ， 也 很 可 能 会 国 为 突然 断 网 、 咖 罪 小 到 服务 器 上 ， 
或 某 个 冒失 鬼 在 错误 的 UNIX 机 器 上 执行 了 shutdown now 命 令 ( 本 书 作 者 之 一 亲身 经 历 
的 著名 事件 ) 等 诸多 原因 而 出 错 。 尽 管 API 的 某 些 方法 还 是 会 偶尔 抛 出 个 Runtime- 
Exception， 但 某 些 开 带 状况 可 以 由 Files .exists (Path) 这 样 的 辅助 方法 来 痢 解 。 


学 习 新 API 最 好 的 办 法 就 是 读 写 代码 。 接 下 来 我 们 来 看 一 些 实际 案例 ， 先 从 基本 的 文件 创建 
和 删除 开始 。 


2.4.1 创建 和 删除 文件 


只 需要 调用 Files 类 里 的 辅助 方法 ， 就 可 以 很 容易 地 创建 和 删除 文件 。 当 然 ， 你 接 到 的 任务 
不 可 能 总 像 默 认 情 况 那 么 简单 ， 所 以 我 们 额外 加 了 一 些 选项 ， 比 如 在 新 创建 的 文件 上 设 定 可 读 / 
可 写 / 可 执行 的 安全 访问 权限 。 


提示 如果 你 要 在 自己 的 机 器 上 运行 本 节 中 的 代码 ， 请 用 实际 路 径 赫 换 掉 代码 中 的 路 径 。 


下 面 的 代码 展示 了 基本 的 文件 创建 操作 ， 用 到 了 Files.createFile(Path target) 方 法 。 
如 果 你 的 操作 系统 里 有 个 D"\Backup 目 录 ， 运 行 代码 之 后 就 会 在 那里 创建 一 个 MyStuff.txt 文 件 。 


Path target = Paths.get ("D:\\Backup\\MyStuff .txt"):; 
Path file = Files.createFile(ltarget}.; 


通常 出 于 安全 考虑 ， 要 定义 所 创建 的 文件 是 用 于 读 、 写 、 执 行 ， 或 三 者 权限 的 某 种 组 合 时 ， 
你 要 指明 该 文件 的 某 些 FileAttributes。 因为 这 取决 于 文件 系统 ， 所 以 需要 使 用 与 文件 系统 相 
关 的 文件 权限 类 。 
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下 面 是 一 个 在 POSIX 文 件 系统 "上 为 属 主 、 属 主 组 内 用 户 和 所 有 用 户 设置 读 / 写 许 可 的 例子 。 
这 种 方法 允许 所 有 用 户 对 即将 创建 的 文件 D'\Backup\MyStuff txt 进 行 读 写 操作 。 
Path target = Paths .get ("D:\\Backup\\MyStuff.txt"); 
Set<PosgsixFilePFermission> Perms = 
PosixFilePermissions.fromString("rw-rw-rw-"); 
FilehAttribute<Set<PosixFilePermissgsion>» attr = 
FogixFilePermiseaiona.aaFileAttribute permael)} 
Files.createrFileltarget, attr):; 


在 ] ava.nio.file.attribute 包 里 有 一 大 申 已 经 写 好 的 *FilePermission 类 。 对 文件 属 
性 的 支持 在 2.4.3 节 中 还 有 更 详细 的 论述 。 


敬告” 如果 在 创建 交 件 时 要 指定 访问 许可 ， 不 要 和 忽略 其 父 目 录 强 加 给 该 文件 的 umask 限 制 或 受 
限 许可 。 上 比如 说 ， 你 会 发 现 即 便 你 为 新 文件 指定 了 rw-rw-rw 许 可 ,但 由 于 目录 的 掩 码 ， 
实际 上 文件 最 终 的 访问 许可 却 是 rw-r--r--。 

删除 文件 要 简单 - -此 g 可 以 用 Fi les.deletel(lPath) 方法 ss 下 面 的 代码 删除 了 刚刚 创建 的 

D'BackupNMyStu 人 txt 文 件 。 当 然 ， 运 行 这 个 Java 程 序 的 用 户 需 要 有 删除 文件 的 权限 。 

Path target = Paths.get ('"D:\\Backup\ \MyStuff .txt");} 
Files.delete (target); 


接 下 来 你 将 学 到 如 何在 文件 系统 中 复制 和 移动 文件 。 


2.4.2 文件 的 复制 和 移动 


使 用 Files 类 中 简单 的 辅助 方法 可 以 很 轻松 地 气 成 文件 的 复制 和 移动 。 
下 面 的 代码 演示 了 如 何 用 Files.copy(Path source，Path target) 方 法 完成 基本 的 复 
制 操作 。 


Path source = Paths.get ("Cc:\\My Documents\\Stuff.txt"); 
Path target = Paths.get ("D:\\Backup\\MyStuff .txt"), 
Files.copy (SouUrce, target); 


复制 文件 时 通常 需要 设置 某 些 选 项 。 下 和 面 这 个 例 于 用 到 了 烤 闵 即 灶 挽 已 有 文件 的 选项 。 
import static java.nio.file.StandardCopyoOprtion.«*; 


Path source = Paths.get ("CcC:\\My Documents\\Stuff.txt"),; 
Path target = Paths.get{("D:\\Backup\\MyStuff .txt"); 
Files.copy (source, target, REPLACE EXISTING).; 


其 他 的 复制 选项 包括 COPY_ATTRIBUTES (复制 文件 属性 ) 和 ATOMIC_MovE( 确保 在 两 边 的 
操作 都 成 功 ， 否 则 回 滚 )。 

移动 和 复制 很 像 ， 都 是 用 原子 Files.move (Path source，Path target) 方 法 完成 的 。 
通常 在 移动 文件 时 ， 你 想 要 用 复制 选项 ， 此 时 便 可 以 用 Files .move(Path source, Path 


由 可 移植 操作 系统 接口 (UNIX )， 是 一 种 许多 操作 系统 都 支持 的 基本 标准 。 
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carget ，Copyoptions . . .) 方 法 ， 但 要 注意 变 参 的 使 用 。 
在 下 面 这 个 例子 中 ,我 们 要 在 移动 源 文件 时 保留 其 属性 ,并 且 覆 盖 目 标 文件 ( 如 果 存 在 的 话 )。 
import static java.nio.file.StandardCopyoOption,.*; 

Path scource = Patha.get ("CcC:\\My Documents\\Stuff .txt"),; 

Path target = Paths.get ("D:\\Backup\\MyStuff .txt"); 


Files.move (lsource, target, REPLACE EXISTING, COPY ATTRIBUTES):; 
现在 你 已 经 能 创建 、 删 除 、 复 制 和 移动 文件 了 ， 下 面 该 认真 研究 一 下 Java 7 对 文件 属性 的 支 
持 了 。 


2.4.3 ”文件 的 属性 


文件 属性 控制 着 谁 能 对 文件 做 什么 ,一 般 情况 下 ， 做 什么 许可 包括 能 否 读 取 、 与 人 或 执行 文 
件 ， 而 由 谁 许可 包括 属 主 、 群 组 或 所 有 人 。 

本 节 从 讨论 文件 的 基本 属性 组 开始 ， 比 如 文件 最 后 访问 时 间 以 及 它 是 目录 还 是 符号 链接 等 。 
本 节 的 第 二 部 分 讨论 对 特定 文件 系统 的 文件 属性 的 支持 , 因为 不 同 的 文件 系统 都 有 它们 自己 的 属 
征集 和 属性 含义 的 解释 ， 所 以 这 部 分 比较 难 。 

让 我 们 先 从 了 解 Java 7 对 基本 文件 属性 的 支持 开始 吧 。 

1. 基本 文件 属性 支持 

真正 通用 的 文件 属性 并 不 多 ,但 确实 有 一 组 大 多 数 文 件 系统 都 支持 的 属性 。 接 口 Basic- 
FileAttributes 定 义 了 这 个 通用 集 , 但 实际 上 工具 类 Files 就 可 以 回答 与 文件 相关 的 各 种 问 
题 ， 比 如 下 面 这 些 . 

口 最 后 修改 时 间 是 什么 时 候 ? 

口 它 有 多 大 ? 

口 它 是 符号 连接 吗 ? 

口 它 是 目录 吗 ? 

代码 清单 2-4 说 明了 Files 类 中 用 于 收集 这 些 基 本 文件 属性 的 方法 。 代 码 输出 了 /usr/bin/zip 的 
相关 信息 ， 你 看 到 的 输出 应 该 和 下 面 的 类 似 : 

/usr/bin/zip 

2011-07-20T16:50:182Z 

351872 

false 

false 

{lastModifiedTime=2011-07-20T16:50:192, 

fileKey= (dev=e000002,ino=30871217}, isDirectory=false, 

lastAccessTime=2011=-06-13T23:31:112, isOther=falgse, 


isSymbolicLink=false, isReqularFile=true, 
creationTime=2011-07-20T16:50:182, Size=351872)} 


注意 ， 所 有 这 些 属性 都 是 调用 Files .readattributes(Path path, Stringattributes, 
LinkOption... options) 得 到 的 。 代 人 码 清 单 2-4 如 下 所 示 : 
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代码 清单 2-4 通用 的 文件 属性 
try 
| : 获取 path 
Path zip = Paths.get ("/usr/bin/zip")., < 二 a 
Syveaetem.out .println(Files.gqetbLastModifiedT1ime (£1B)); 


system.out .println(lFiles.size (lzip)): 输出 属性 
System.out .println(Files. 1gsSymbol1icl i (LD) ; 
System.out .print lnlFiles.isDlirectory (21p)):; 


System,out .printlnlFiles. ee ph We 村 
| 和 , 
A 执行 批量 读 取 
catcn |IQOExcCceptlion ex) 


Svystem,.out .println(l"Exception [" + ex.getMessage() + "™]") 

还 有 一 些 可 以 从 Files 类 的 方法 中 采集 到 的 通用 文件 属性 信息 。 这 样 的 信息 包括 文件 属 主 ， 
是 否 为 符号 链接 等 。 请 参照 iles 类 的 Javadoc 查 看 完整 的 辅助 方法 列表 . 

Java 7 也 文 持 跨 文件 系统 的 文件 属性 查看 和 处 理 功能 

2. 特定 文件 属性 支持 

在 2.4.1 节 创建 文件 时 你 已 经 见 过 FileAttribute 接 口 和 PosixFilePermissions 类 了 ,为 

支持 文件 系统 特定 的 文件 属性 ，Java 7 允许 文件 系统 提供 者 实现 FileattributeView 和 

BasicFileattributes 接 口 。 


敬告 我 们 之 前 已 经 说 过 了 ， 但 有 必要 再 重复 一 次 在 编写 特定 文件 系统 的 代 而 时 一 定 要 小 心 
一 于 要 确保 你 的 还 生 和 本 处 理 考 夺 了 代 册 在 不 同 江 件 条 加 上 十 行 的 全 和 


一 一 


EE a 


来 看 其 中 你 想 用 Java 7 保 让 正确 的 访问 主 可 蕉 po A 中 。 图 2-2 显 示 了 
Admin 用 上 di | 录 的 列表 。 注 意 那个 特殊 的 ,profile 隐 藏 文件 ， 中 个 许 Admin 用 户 写 ， 其 他 
任何 人 都 没有 与 权限 ， 但 了 所 有 人 虱 可 以 计 取 该 文件 ， 


图 2-2 ”Admin 用 尸 的 home 目 录 列 表 ， 显 示 .profile 的 访问 许可 


在 下 面 的 代码 中 ,你 要 确保 .profile 文 件 的 访问 许可 设置 正确 ,与 图 2-2 对 应 。Admin 用 户 希 望 
其 他 所 有 用 户 都 可 以 读 取 该 文件 ， 但 只 有 他 自己 来 写 。 你 可 以 用 特定 的 POSIX PosixFile- 
Permission 利 PosixFilLeattributces 关 来 你 证 访问 许可 (rw-r--r--) 是 正确 的 ] 。 


只 自 在 读书 @ 


WWW .Zizidiary .com 
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代码 清单 2.5 Java 7 对 文件 属性 的 支持 


import static java.nio.file.attribute.PosixFilePermission.™:; 


try 
Path profile = Paths.get ("/user/Admin/ .profile™"),; 获取 属性 视图 
POSIXFileAttributes attrs = 
Files.readAttributeasl(profile, 


PogixFileAttributes.class); ， 
? 读 取 访问 许可 
Set<PosixFilePermission> posixPermisaions = 


attrs.permissions'():; 


posixPermissions,clear(); 二 
String owner = attrs.owner() .getName (); © 消除 访问 许可 


String perms = 日 志 信息 
PosixFi]JePpermissions.toString (posixPpermissions); i 
System,out .format ("Ys $sten", OWnNner, perms):; 


posixPermissions.add (OWNER READ) :; 

posixPermissions.add (GROUP READ); MOE 
posixPermissions.add (OTHER READ); KD 设置 新 的 访问 许可 
posixPermissions.add (OWNER WRITE) : 

Files.setPosixFilePermlissions lprofile, posixPermissions),; 


| 


catch (IOExcept ion ) 

| System.out .println (e.getMessage ()); 

代码 从 导 人 PosixFilepPermission 常 量 还 有 其 他 未 显示 的 导 人 开始 ， 然后 得 到 .profile 文 件 
的 Path。Files 类 中 有 个 辅助 方法 让 你 可 以 读 取 特定 文件 系统 的 属性 ， 在 这 个 例子 中 是 
PosixFileAttributes 偶 。 然 后 你 就 可 以 访问 PosixFilePermission 全 ,在 清除 了 已 有 的 许 
可 之 后 合 ， 你 可 以 为 文件 添加 新 的 访问 许可 ， 妆 然 还 是 用 Files 中 的 方法 人 @。 

你 可 能 已 经 注意 到 了 ，PosixFilePermission 是 一 个 enum,， 内 此 没有 实现 FileAttri- 
buteView 接 口 。 为 什么 这 里 没 用 posixFileaAttributeview 呢 ?实际 上 是 Files 辅 助 类 把 它 隐 
藏 了 起 来 这样 你 就 可 以 用 readAttributes 方 法 直接 读 取 文件 属性 了 ， 也 可 以 用 setPosix- 
FilePermissions 方 法 和 直接 设置 访问 许可 。 

除了 基本 属性 ，Java 7 还 有 一 个 用 来 支持 特别 操作 系统 特性 的 扩展 系统 。 可 惜 ， 我 们 不 可 能 
者 括 所 有 特殊 情况 ， 但 我 们 会 给 你 看 一 个 扩展 系统 的 例子 : Java 7 对 符号 链接 的 支持 。 

3. 符号 链接 

你 可 以 把 符号 链 接 看 做 指 辣 男 一 个 文件 或 目录 的 入 口 ， 并 且 在 大 多 数 情 况 下 它们 都 是 透明 
的 。 比 如 切换 到 符号 链接 的 目录 下 会 把 你 带 到 符号 链接 所 指向 的 目录 下 。 但 在 写 软件 时 ， 比 如 备 
份 工具 或 部 着 脚本 ， 你 需要 慎重 考虑 是 否 应 该 跟随 符 导 链接 ，NIO.2 人 允许 你 做 出 选择 。 

我 们 再 用 一 下 2.2.3 节 的 例子 。 你 要 在 *nix 系 统 上 查询 /usr/logs 目 录 下 的 日 志文 件 log1.txt 的 信 
息 。 但 /usrlogs 目 录 实 际 上 是 一 个 指向 /application/logs 目 录 的 符号 链接 ( 指针 )，/application/logs 
目录 才 是 日 志文 件 的 真正 位 置 。 
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符号 链接 在 宿主 操作 系统 中 使 用 ,包括 ( 但 不 限于 ) UNIX、Linux、Windows 7 和 Mac OS X。 
Java 7 对 符号 链接 的 支持 遵循 UNIX 操 作 系 统 中 实现 的 语义 。 

下 面 的 代码 在 读 取 基本 文件 属性 之 前 先 检查 指向 安装 Java 的 /optplatform 目 录 的 Path， 看 它 
是 否 为 符号 链接 ， 我 们 想 读 取 文件 真正 位 置 的 属性 。 代 码 清单 2-6 如 下 所 示 ， 


Path file = Paths.get("/opt/platform/java"):; 由 检查 符号 链接 
try 
| 
if (Files.isSymbolicLink (file)) 9 读 取 符号 链接 
| 
file = Files.readSymbolicLink (file).; 


| 


Files.readAttributes(file, BasicFileAttributes.class),) ; 
| 
读 取 文件 属性 


catch {IOException €) 


System.out .printlnle.getMessage()}; 


Files 类 提供 了 一 个 issymbolicLink (Path) 方 法 来 检查 符号 链接 加。 它 还 有 一 个 辅助 方 
法 ， 可 以 用 于 返回 符号 链接 目标 的 真实 Path@@， 所 以 你 能 读 到 正确 的 文件 属性 合 。 

NIO.2 API 默 认 会 跟随 符号 链接 。 如 果 不 想 跟随 ， 震 要 用 Linkoption.NOFOLLOW_LINKS 选 
项 , 这 一 选项 可 以 用 在 几 个 方法 调用 上 .。 如 果 你 要 读 取 符号 链接 本 身 的 基本 文件 属性 , 应 该 调用 : 


Files.readAttributes(target, 
BasicFileAttributes.class, 
LinkOption.NOFOLLOW LINKS) ; 


香 号 链接 是 Java 7 对 特定 文件 系统 支持 最 常用 的 例子 ,API 设 计 者 也 考虑 到 了 未 来 对 特定 文件 
系统 支持 特性 的 扩展 ， 比 如 量子 加 密 文 件 系统 。 
你 已 经 做 过 文件 处 理 了 ， 现 在 可 以 开始 研究 对 文件 内 容 的 处 理 了 。 


2.4.4 快速 读 写 数据 


Java 7 可 以 尽 可 能 多 地 提供 用 来 该 取 和 写 人 文件 内 容 的 辅助 方法 。 当 然 ， 这 些 新 方法 使 用 
Path， 但 它们 也 可 以 与 那些 java.io 包 里 基于 流 的 类 进行 互 操作 。 因 此 ， 你 用 一 个 方法 就 可 以 
读 取 文件 中 的 所 有 行 或 全 部 字 节 。 

本 节 会 向 你 介绍 打开 文件 ( 带 选 项 ) 的 过 程 ， 以 及 一 小 组 常用 的 文件 读 / 写 例子 。 让 我 们 先 
从 打开 文件 的 不 同方 式 开始 。 

1. 打开 文件 

Java 7 可 以 直接 用 囊 缓 冲 区 的 读 取 器 和 写 人 器 或 输入 输出 流 ( 为 了 和 以 前 的 Java LO 代码 兼 
容 ) 打开 文件 。 下面 的 代码 演示 了 Java 7 如 何 用 Files .newBufferedReader 方 法 打开 文件 并 按 
行 读 取 其 中 的 内 容 。 


只 自 在 读书 @ 
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Fath logFile = Pathas.get("/tmp/app.109g"); 
try (BufferedReader reader = 
Files.newBufferedReader(logFile, StandardCharsets.UTF 8)) | 
String line: 
while [lline = reader ,readLinel()) != null) | 
| 
打开 一 个 用 于 写 人 的 文件 也 很 简单 。 
Fath logFile = Paths.get ("/tmp/app .Leg") ; 
try (BufferedWriter writer = 
Files.newBufferedWrite (logFile, StandardcCcharsetsas.UTF 8, 
Scandardopenoption.WRITE)) 1 
Writer .write("Helle Worldl!l"):; 
} 
注意 standardOpenOption .WRITE 选项 的 使 用 ， 这 是 可 以 添加 的 几 个 openoption 变 参 之 
一 。 它 可 以 确保 写 入 的 文件 有 正确 的 访问 许可 。 其 他 常用 的 文件 打开 选项 还 有 READ 和 APPEND。 
与 Inputstream 和 Outputstream 的 交互 是 通过 Files .newInputstream(Path,OpenOption...) 
和 Files .newoutputstream(Path,OpenOption...) 实 现 的 。 它 们 为 过 去 基于 java .io 包 的 
LIUO 和 新 的 基于 java.nio 包 的 文件 UO 之 间架 起 了 一 座 桥 梁 。 


提示 。 在 处 理 String 时 ， 不 要 忘 了 查看 它 的 字符 编码 。 忘 记 设 置 字符 编码 ( 通过 standard- 
charsets 类 ， 比 如 new String(byte[],StandardCcharsets.UTF_8)) 可 能 寺 致 不 可 
预料 的 字符 编码 问题 。 


前 面 的 代码 片段 还 是 用 Java 6 及 之 前 版 本 编写 的 读 取 和 写 人 文件 代码 ， 仍 然 属于 比较 繁 珊 的 
底层 代码 。 而 Java 7 具备 更 高 层 的 抽象 能 力 ， 可 以 帮 你 避免 很 多 不 必要 的 繁琐 编码 工作 。 

2. 简化 读 取 和 写 入 

辅助 类 Files 有 两 个 辅助 方法 ， 用 于 读 取 文件 中 的 全 部 行 和 全 部 字 节 。 也 就 是 说 你 没 必 要 再 
用 whi1le 循 环 把 数据 从 字 节 数组 读 到 缓冲 区 里 去 。 下 面 的 代码 演示 了 如 何 调 用 辅助 方法 。 


Path logFile = Paths.get ("/tmp/app.109g"):} 
List<String> lines = Files.readAllLinesa(logFile, StandardCharsets.UTF 8); 
byte[] bytes = Files.readAllBytes (llogFile).; 


对 于 某 些 软 件 来 说 , 什么 时 候 读 、 写 是 个 问题 ,特别 是 在 处 理 属性 文件 或 日 志 时 。 这 时 就 该 
文件 修改 通知 系统 大 显 身 手 了 。 


2.4.5 ”文件 修改 通知 


在 Java 7 中 可 以 用 java.nio.file.Watchservice 类 监测 文件 或 目录 的 变化 。 该 类 用 客户 
线程 监视 注册 文件 或 目录 的 变化 , 并 且 在 检测 到 变化 时 返回 一 个 事件 。 这 种 事件 通知 对 于 安全 监 
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测 、 属 性 文件 中 的 数据 刷新 等 很 多 用 例 都 很 有 用 。 是 现在 某 些 应 用 程序 中 常用 的 轮 询 机 制 ( 相对 
而 言 性 能 较 差 ) 的 理想 替代 品 。 

下 面 的 代码 用 watchservice 监 测 用 户 karianna 主 目录 的 变化 ， 每 当 发 现 变化 时 就 会 在 控制 
台中 输出 一 个 事件 通知 。 和 很 多 持续 轮 询 的 设计 一 样 , 它 也 需要 一 个 轻 量 的 退出 机 制 。 代 码 清单 
2-7 如 下 所 示 : 
代码 清单 2-7 使 用 Watchservice 

import static java.nio.file.StandardWatchEventKinds.*; 

by 


WatchService watcher = 
FileSsystems .getDefault () .newwatchService1()， 监测 变化 


Path dir = 
FileSystems .getDefault () .getPath|("/usr/karianna"); 


WatchKeY key = dir.register (watcher, ENTRY MODIEFY) ; 
whilel!shutdown) = 一 从 检查 shutdown 标 志 
| 
key = Watcher.takel):; 得 到 下 一 个 key 


for (WatchEvent<?> event: key.pollEvents'()) : 及 其 事件 
| 
if levent .kind() == ENTRY MODIFY) J 
System,.out .printlnl"Home dir changed!"),; 检查 是 否 为 
| 变化 事件 
| 


key, reset () ; 二 一 一 一 人 重 置 监测 key 


catch (IOException | InterruptedException e) 

Syatem,out .printilnle.getMeaasage()); 

| 

在 得 到 默认 的 watchservice 后 ， 将 karianna 的 主 目录 登记 到 变化 监测 名 单 中 塌 。 然 后 在 一 
个 无 限 循环 ( 直到 shutdown 标 志 改 变 ) 动 中 执行 Watcherservice 的 take{) 方 法 ， 直到 WatchKey 
的 到 来 ,一 旦 得 到 watchkey, 代码 就 遍历 其 watchEvent 进 行 检测 全 . 如 果 发 现 了 类 型 为 ENTRY- 
MODIFY 的 WatchEvent 信 ,就 诏 告 天 下 karianna 的 主 目录 发 生 了 变化 ! 最 后 重 置 key 合 准备 迎接 下 
一 个 事件 ， 继 续 等 待 。 

还 有 其 他 可 以 监测 的 事件 ， 比 如 ENTRY_CREATE、ENTR 
事件 已 经 丢失 或 被 丢弃 了 )。 

接 下 来 ,我 们 要 进 人 一 个 非常 重要 的 、 抽 象 的 新 API 一 一 用 于 数据 的 读 写 ,使 异步 LO 成 为 现 
实 的 SeekabLeBytechanne1l。 


Y_DELETE 利 OVERFLOW ( 可 以 表明 
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24.0 SeekableByteChannel 


Java 7 引 人 SeekablIeBytechanne1l 接 口 ， 是 为 了 让 开发 人 员 能 使 改变 字 节 通道 的 位 置 和 大 
小 。 比 如 ， 应 用 服务 器 为 了 分 析 日 志 中 的 某 个 错误 码 , 可 以 让 多 个 线程 去 访问 连接 在 一 个 大 型 日 
志文 件 上 的 字 节 通道 。 

JDK 中 有 一 个 java.nio.channels.SeekableBytechannel 接 口 的 实现 类 一 一 java.nio. 
channels.Filechanne1l。 这 个 类 可 以 在 文件 读 取 或 写 人 时 保持 当前 位 置 。 比 如 说 ， 你 可 能 想 
要 与 一 段 代 码 谈 取 日 志文 件 中 的 最 后 1000 个 字符 ,或 者 向 一 个 文本 文件 中 的 特定 位 置 与 人 一 些 价 
格 数据 。 

下 面 的 代码 展示 了 如 何 运用 Filechannel 的 寻 址 能 力 读 取 日 志文 件 中 的 最 后 1000 个 字符 。 


Path logFile = Paths.get("c:\\temp.10g"); 

ByteBuffer buffer = ByteBuffer.allocate (1024).,; 

FileChannel channel = FileChannel .open(llogFile, StandardoOpenOption.READ); 
channel .read (buffer, channel .size() - 1000); 


Filechannel 类 的 寻 址 能 力 意 味 着 开发 人 员 可 以 更 加 灵活 地 处 理 文 件 内 容 。 我 们 期 待 能 由 此 
产生 一 些 有 趣 的 开源 项 目 ， 比 如 针对 大 型 文件 的 并 行 访问 。 随 着 对 该 接口 的 不 断 扩 展 ， 可 能 还 会 
有 网 络 数据 流 的 续 传 。 

NIO.2API 中 下 一 个 主要 修改 是 异步 IJD， 它 可 以 使 用 多 个 后 台 线 程 读 写 文 件 、 套 接 字 和 通道 
中 的 数据 。 


2.5 异步 I/O 操作 


NIO.2 另 一 个 新 特性 是 异步 能 力 ， 这 种 能 力 对 套 接 宇 和 文件 IO 都 适用 。 异 步 JO 其 实 只 是 一 
种 在 读 写 操作 结束 前 允许 进行 其 他 操作 的 LO 处理。 实际 上 ， 就 是 可 以 充分 利用 最 新 的 硬件 和 软 
件 特性 ， 比 如 多 核 CPU 及 操作 系统 对 套 接 字 和 文件 处 理 的 支持 。 对 于 任何 想 在 服务 器 端 和 系统 级 
编程 领域 占有 一 席 之 地 的 编程 语言 来 说 ,异步 JO 都 是 必 不 可 少 的 特性 。 我 们 相信 ，Java 在 服务 器 
问 编 程 培 言 中 所 取得 的 重要 地 位 会 因为 该 特性 得 以 延续 。 

举 个 简单 的 例子 ,想象 一 下 你 要 把 100GB 的 数据 写 人 文件 系统 或 网 络 套 接 字 中 。 如 果 你 用 的 
是 老 版 本 的 Java， 在 同时 把 数据 写 人 文件 或 套 接 字 的 多 个 区 域 时 ， 必 须 亲 自动 手 用 
java.util.concurrent 写 多 线程 代码 。 当 然 ， 同 时 进行 多 路 读 取 也 不 容易 。 除 非 你 写 的 代码 
十 分 巧妙 , 和 理 则 在 使 用 LO 时 也 会 阻塞 主线 程 ,这 意味 着 在 你 完成 漫长 的 VO 操作 之 前 , 除了 等 待 ， 
还 是 等 待 。 


提示 “如 果 你 还 没 接 触 过 NIO 通 道 ， 也 许可 以 趁 此 机 会 充实 下 你 的 知识 结构 。 不 过 这 个 领域 的 
新 内 容 很 少 ， 但 在 继续 本 节 内 容 之 前 ， 狐 们 建议 你 去 看 看 Ron Hitchens 写 的 Java NIO 
【O"Reilly，2002 ) 一 书 ， 你 会 从 中 诸 益 匪 浅 。 


1 入 
Yr 


38 第 2 草 新 LO 


Java 7 中 有 三 个 新 的 异步 通道 : 

口 AsynchronousFileCchannel1 一 一 用 于 文件 1/O，; 

口 AsynchronousSocketChannel 一 一 用 于 套 接 字 WO， 支 持 超 时 ; 

口 AsyvnchronousServerSocketchanne1 一 一 用 于 套 接 字 接 受 异 步 连 接 。 

使 用 新 的 异步 WO API 时 ， 主 要 有 两 种 形式 ,将 来 式 和 回调 式 。 有 趣 的 是 ， 这 些 异步 API 用 到 
了 第 4 章 讨 论 的 一 些 现代 并 发 技术 ， 所 以 这 真是 让 你 先睹为快 了 。 

我 们 会 从 异步 文件 访问 的 将 来 式 开 始 。 希望 你 已 经 用 过 这 种 并 发 技术 , 但 如 果 没 有 用 过 , 也 
不 用 担心 ， 本 节 将 会 讲解 得 非常 详细 ， 即 便 是 刚 接触 这 个 话题 的 新 手 也 能 看 明白 。 


2.5.1 将 来 式 


NIO.2 API 的 设计 人 员 用 将 来 式 ( future ) 这 个 术语 来 表明 使 用 java .util .concurrent. 
Future 接 口 。 当 你 布 望 由 主 控 线 程 发 起 IO 操作 并 轮 询 等 待 结果 时 , 一 般 都 会 用 将 来 式 异 步 处 理 。 

将 来 式 用 现 有 的 java.util.conecurrent 技 术 声 明 一 个 future, 用 来 保存 异步 操作 的 处 理 
结果 。 这 很 关键 ， 因 为 这 意味 着 当前 线程 不 会 因为 比较 慢 的 IO 操作 而 停 河 。 相 反 ， 有 一 个 单独 
的 线程 发 起 UVO 操 作 ， 并 在 操作 完成 时 返回 结果 。 与 此 同时 ， 主 线程 可 以 继续 执行 其 他 需要 完成 
的 任务 。 在 其 他 任务 结束 后 ， 如 果 UO 操 作 还 没有 完成 ， 主 线程 会 一 直 等 待 。 图 2-3 演 示 了 一 个 用 
将 来 式 读 取 大 型 文件 的 过 程 。( 代码 清单 2-8 是 相应 的 实现 代码 。) 

主线 程 异步 处 理 


Paths .get (filename) 


Asynchronouari laeChannel 


ByteBuffer 


| 
上 
和 


MAsynchronousFileCchannel 


isDone i{) 二 


机 画面 量 - 轩 


图 2-3 ”将 来 式 异 步 读 取 
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通常 会 用 Future get() 方 法 ( 带 或 不 带 超时 参数 ) 在 异步 LO 操作 完成 时 获取 其 结果 。 假 
设 你 要 从 硬盘 上 的 文件 里 读 取 100 000 个 字 节 ,在 旧版 的 Java 中 ,你 需要 等 待 数据 读 取 完成 ( 除非 
你 实现 了 一 个 线程 池 ， 而 且 工 作 线程 使 用 java .util .concurrent 技 术 ， 这 可 不 是 件 轻 松 的 事 
儿 )。 而 在 Java 7 中 ， 主 线程 可 以 在 读 取 数据 的 同时 继续 完成 其 他 工作 ， 如 下 面 的 代码 所 示 。 


代码 清单 2-8 ”异步 IO 一 一 将 来 式 


try 

| 
Path file = Paths.get ("/usr/karianna/foobar .txt")., 
AasynchronousFileChannel channel = 局 异步 打开 文件 
AaynchronouseFilecCchannel .open (file),; | 


ByteBuffer buffer = ByteBuffer.allocate(100 000) ; 局 读 取 100 000 字 节 
Future<Integer> result = channel .read (buffer, 0),; 


whilel!lresult .isDonel()) pi 干 点 儿 别 的 事情 

| 为 中 日 | 
ProfitCalculator .calculateTax(}:; 

) 1 获取 结果 

Integer bytesRead = result .get ();} 


System.out .println("Bytes read [" + bytesRead + "]"); 
| 
catch (IOException | ExecutionException | InterruptedException e) 
system.out .printlnle.getMessage()); 
| 
上 面 的 代码 一 开始 先 用 后 台 进 程 中 打开 一 个 AsynchronousFileChannel 读 /与 foobartxt@@， 
接 下 来 的 这 一 步 是 为 了 让 UVO 处 理 能 跟 发 起 它 的 线程 同步 进行 。 因 为 采用 Asynchronous- 
Filechannel， 并 用 Future 保 存 读 取 结果 ， 所 以 会 自动 采用 并 发 的 O 处 理 和 四。 在读 取 数 据 时 ， 
主线 程 可 以 继续 执行 任务 比如 算 一 下 要 交 多 少 税 ) @。 最 后 ， 当 任务 完成 时 ， 你 可 以 检查 数据 


一定 要 注意 ， 我 们 在 这 里 用 ispone() 手 工 判 定 result 是 否 结 束 。 通 常情 况 下 , result 或 结束 ( 主 
线程 会 继续 执行 )， 或 等 待 后 从 1/0 完 成 。 

你 可 能 会 好 奇 这 究竟 是 怎么 实现 的 。 长 话 短 说 ，APLJVM 为 执行 这 个 任务 创建 了 线程 池 和 通 
道 组 。 另外， 你 也 可 以 自己 提供 和 配置 一 个 。 解 释 其 中 的 细节 顾 费 口舌 ， 并 且 官 方 文档 都 解释 过 
了 ， 所 以 我 们 只 是 直接 引用 了 AsvnchronousFilechannel 的 Javadoc: 

RSynchronousFilechanne1l 会 关联 线程 池 ， 它 的 任务 是 接收 JJD 处 理事 件 ， 并 分 发 给 负责 
处 理 通 道中 IO 操作 结果 的 结果 处 理 嚣 。 跟 通道 中 发 起 的 IO 操作 关联 的 结果 处 理 器 确保 是 由 线程 
池 中 的 某 个 线程 产生 的 。 

如 果 在 创建 asvnchronousFilechannel 时 设 有 为 其 指明 线程 池 , 那 就 会 为 其 分 配 一 个 系统 
揪 认 的 线程 池 ( 可 能 会 和 其 他 通道 共享 )。 默 认 线程 池 是 由 AsynchronouschannelGroup 类 定 
义 的 系统 属性 进行 配置 的 。 
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此 外 还 有 一 种 被 称 为 回调 的 技术 。 有些 开 发 人 员 可 能 会 发 现 回调 式 用 起 来 更 方便 ,因为 它 很 
像 Swing、 消 息 和 其 他 Java API 中 出 现 过 的 事件 处 理 技术 。 


2.5.2 回调 式 


与 将 来 式 相 反 ， 回 调式 ( callback ) 所 采用 的 事件 处 理 技术 类 似 于 在 Swing UI 编程 时 及 用 的 
机 制 。 其 基本 思想 是 主线 程 会 派 一 个 侦查 员 completionHandler 到 独立 的 线程 中 执行 LO 操作 。 
这 个 侦查 员 将 带 着 WO 操作 的 结果 返回 到 主线 程 中 ， 这 个 结果 会 触发 它 自己 的 completed 或 
failed 方 法 ( 你 会 重 写 这 两 个 方法 )。 

在 异步 事件 刚 一 成 功 或 失败 并 需要 马上 采取 行动 时 , 一 般 会 用 回调 式 。 比 如 在 读 取 对 昔 利 计 
算 业 务 处 理 至 关 重 要 的 金融 数据 时 ,如果 读 取 失败 了 ,你 最 好 马上 就 执行 回 滚 操作 , 或 进行 异常 
处 理 。 

在 异步 /0 活动 结束 后 ， 接 口 java.nio.channels .CompletionHandler<V,A> 会 被 调用 ， 
其 中 V 是 结果 类 型 ，A 是 提供 结果 的 附着 对 象 。 此 时 必须 已 经 有 了 该 接口 的 completed(V,A) 和 
failedfV,RA) 方 法 的 实现 ， 你 的 程序 才能 知道 在 异步 JO 操 作成 功 完 成 或 因 某 些 原因 和 失败 时 该 如 
何 处 理 。 图 2-4 展 示 了 这 一 过 程 〈 代 码 清 单 2-9 是 该 过 程 的 实现 代码 )。 

主线 程 异步 处 理 


Paths.get (filename) 


AsynchronousFileCchanne) 


押 着 CompleticnHandlLer 


reaa 文 件 


取得 结果 


7 结果 


处 央 结 内 -omplered tf 商 卫 儿 呈 局 


图 2-4 回 再 式 异 步 读 取 


在 下 例 中 , 你 又 一 砍 从 foobartxt 文 件 中 读 取 了 100 000 字 节 的 数据 , 用 completionHandler< 
Tnteger ,ByteBuffer> 声 明 是 成 功 或 是 失败 。 
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Path file = Paths.get("/usr/karianna/foobar.txt"); 


hasynchnronousFileChannel channel = 以 凑 步 方 洋 打 开交 由 
AsynchronousFileCchannel .open(lfilel.; , i 


ByteBuffer buffer = ByteBuffer.allocate (100 000) ) 
channel .readlbuffer, 0, buffer., | 1 
new CompletionHandler<Integer, ByteBuffer>() 从 通道 中 读 取 数据 


Public void completed'lInteger result, 
ByteBuffer attachment) 
{ 读 取 完成 时 的 


Syatem.out .printlnl"Bytes read [" + result + "]"); 回调 方法 


public void failed(Throwable exception, ByteBuffer attachment) 
System.out .printlnlexception.getMessage'(')}); 


} 
让 
(IOException 已 
System.out .printlnle.getMessage!|)); 
| 
本 节 中 的 两 个 例子 都 是 基于 文件 的 ， 但 将 来 式 和 回调 式 腊 步 访问 也 适用 于 Asynchronous- 
ServerSocketchannel 和 AsynchronousSocketCchannel。 开 发 人 员 可 以 用 它们 编写 程序 来 外 
理 网 络 套 接 字 ， 比 如 语音 卫 或 写 出 性 能 更 优异 的 客户 痊 和 服务 器 端 软件 。 
接 下 来 的 一 系列 变化 统一 本 套 接 字 和 通道 ,让 你 可 以 将 套 接 宇和 通道 区 互 的 管理 归结 到 AP[ 中 。 


31 的 整合 


应 用 软件 对 网 络 接 人 的 需求 比 以 往 任 何 时 候 都 要 迫切 。 仿佛 一 夜 之 间 , 家 里 所 有 东西 都 要 联网 
了 。 在 旧版 Java 中 ， 套 接 字 和 通道 结合 得 并 不 是 很 好 ， 将 它们 两 个 配合 在 一 起 是 件 环 手 的 事情 。 因 
此 Java 7 推出 了 Networkchannel， 把 socket 和 channel 结 合 到 一 起 ， 让 开发 人 员 可 以 轻松 应 对 。 
编写 底层 网 络 代 码 算 是 专业 领域 。 如 果 你 的 工作 领域 与 此 无 美 , 完全 可 以 跳 过 这 一 节 ! 但 如 
果 怡 好 你 就 是 干 这 个 的 ， 你 可 以 在 本 节 对 Java 7 的 新 特性 有 一 个 初步 了 解 。 
我 们 先 来 看 看 套 接 字 和 通道 在 Javadoc 中 的 定义 ， 重 温 一 下 它们 在 Java 中 扮演 的 角色 : 
java.nio,.,channels 包 
定义 通道 ， 表 示 连 接 到 执行 WO 操作 的 实体 ， 比 如 文件 和 套 接 字 。 定 义 用 于 多 路 侍 
输 、 非 阻塞 UO 操 作 的 选择 器 。 
java net . Socket 类 
该 类 实现 了 客户 端 套 接 字 ( 也 称 为 “ 套 接 字 ”)。 套 接 字 是 两 个 机 器 间 通 信 的 端点 。 


2.6 Socket 和 Char 
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在 旧版 Java 中 , 为 了 执行 LO 操作 ， 比 如 向 TCP 并 口中 写 信 数据, 你 需要 将 通道 绑 定 到 socket 
的 实现 类 上 ,但 channel 和 socket 彼 此 之 间 却 有 “代沟 ”， 

口 在 旧版 Java 中 ， 为 了 配置 套 接 宇 选项 和 绑 定 在 套 接 宇 上 ， 必 须 把 ; 

在 一 起 ; 

口 在 旧版 Java 中 ， 不 能 利用 平台 特定 的 套 接 字 行为 。 

让 我 们 来 看 看 新 接口 NetworkChannel 和 其 子 接口 MulticastChannel 对 这 两 个 领域 做 的 
“整理 ”工作 。 


2.06.1 NetworkChannel 


新 接口 java.nio.channels .NetworkChannel 代 表 一 个 连接 到 网 络 套 接 字 通道 的 映射 。 
它 定 义 了 一 组 实用 的 方法 ,比如 查看 及 设置 通道 上 可 用 的 套 接 字 选 项 等 。 下面 的 代码 运用 这 些 广 
法 输出 互联 网 套 接 字 地 址 在 端口 3080 上 所 支持 的 选项 , 设置 IP 服 务 条 款 选 项 以 及 确认 套 接 字 通道 
上 的 so_KEEPALIVE 选 项 。 


代码 清单 2-10 Networkchanne1 选 项 

SelectorProvider provider = SelectorProvider.provider'(); 

try 

| 
NetworkcChannel socketChannel = 

provider .openSsSocketChannel (); 将 Networkchannel 

SocketAddress address = new lnetSocketAddress (3080):; 绪 定 到 端口 3080 上 
socketChannel = socketChannel .bindladdresas).; 


Set<SocketOption<?>»> socketOptions = | 
socketchannel ,Bupportedoptiocongs1() ; 检查 套 接 字 选 项 
System.out .printlnlsocketoptions.toString'())})., 
socket channel .setoOption lstandardSocketOptions.T1TP TOS, 设 置 套 接 字 的 Tog 
Ss | (服务 条 款 ) 沈 


Boolean keepAlive = 
socketChannel .getoOption (StandardSocketOptions .SO KEEBPALIVE).; 


) Ny 获取 gO_KEEPALIVE 
z es 选项 
catch (IOException e€) 


| 


} 
此 外 ，Networkchanne1 的 出 现 使 得 多 播 操作 成 为 可 能 。 


System.out .printlnle.getMessage()).; 


2.0.2 MulticastChannel 


像 BitTorrent 这 样 的 对 等 网 络 程 夺 一 般 都 其 备 多 播 的 功能 。 在 Java 的 早期 版 本 中 , 虽然 拼凑 一 
下 也 能 实现 多 播 ， 但 却 没 有 很 好 的 API 抽 和 象 层 。Java 7 中 的 新 接口 MulticastCchannel 解 决 了 这 
个 问题 。 


1 地 
OEEEE 


2.7 小 结 43 


术语 多 播 ( 或 组 播 ) 表示 一 对 多 的 网 络 通 讯 ， 通 常用 来 指 代 IP 多 播 。 其 基本 前 提 是 将 一 个 包 
发 送 到 一 个 组 播 地 址 ， 然 后 网 络 对 该 包 进 行 复制 ， 分 发 给 所 有 接收 疹 (注册 到 组 播 地 址 中 )， 如 
图 2-5 所 示 。 


图 2-5 多 播 示 例 

为 了 让 新 来 的 Networkchanne1l 加 人 允 播 组 ，Java 7 提供 了 一 个 新 接口 java.nio. 
channels.Multicastchanne1l 及 其 默认 实现 类 Datagramchanne1l。 也 就 是 说 你 可 以 很 轻松 地 
对 多 播 组 发 送 和 接收 数据 。 

下 面 的 代码 说 明了 如 何 加 入 IP 地 址 为 180.90.4.12 的 多 播 组 ,并 对 其 发 送 和 接收 系统 状态 信息 。 
代码 清单 2-11 Networkchanne1 选 项 

try 

artoneemederrnaa erm a 


DatagramChannel de = ] : 
DatagramChannel .open(lSstandardProtocolFamily. INET), 打开 DatagramChannel 


dc .setOption'lStandardSocketOptions.SsoO REUSEADDR., 


truel):; 
dc.bind(lnew InetSocketAddress (8080)):; i 
= 二 而 一 i i / 将 通道 设置 为 多 播 
dc .setoptionlstandardSsocketOptions.1IP MULTICAST IF, 
networkInterface); 
InetAddress group = 
InetAddress.getByName ("180.90.4.12"); ] 加 入 多 播 组 
MembershipkKey key = dc.]joinlgroup, networkIntertface)., 
| 
catch (IOQEXCepticn ee) 
| 


System,.out .printlnle.getMesgagel()); 


到 此 为 止 ， 我 们 对 NIO.2 API 的 初步 研究 已 经 纺 束 了 。 和 希望 你 喜欢 这 次 行 色 勿 匆 的 旅程 ! 


2.7 小结 


硬件 和 软件 LO 的 发 展 突飞猛进 ， 而 Java 7 也 紧 随 其 后 ， 充 分 利用 了 NIO.2 的 新 API。Java 7 提 
供 的 新 类 库 可 以 用 来 处 理 位 置 ( Path )， 用 来 在 文件 系统 上 执行 操作 ， 比 如 处 理 文件 、 目 录 、 符 
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号 链接 等 。 特 别 值得 一 提 的 是 ， 在 平台 特性 的 支持 下 ，Java 7 可 以 任意 穿梭 于 文件 系统 中 ， 并 能 
够 处 理 大 型 目录 结构 。 

NIO.2 致 力 于 为 那些 通常 需要 大 量 编码 工作 的 任务 提供 一 站 式 的 解决 办 法 。 尤 其 是 新 的 File 
工具 类 ， 它 有 很 多 辅助 方法 ， 比 起 原来 的 java.io.File， 它 使 得 编写 文件 IO 代码 更 快 ， 也 更 
简单 。 

异步 10 是 一 个 强大 的 新 特性 ， 可 以 保证 在 处 理 大 文件 时 性 能 不 受到 显著 影响 。 它 对 网 络 套 
接 字 和 通道 流量 异常 繁忙 的 程序 也 很 有 帮助 。 

NIO.2 也 用 到 了 来 自 Coin 项 目 (第 ] 章 ) 的 新 特性 。 这 使 得 在 Java 7 中 处 理 VO 比 以 往 的 版 本 更 
安全 ， 而 且 所 需 的 代码 会 更 少 。 

现在 是 时 候 进 和 本 书 的 第 二 部 分 卫 。 让 你 的 大 脑 准备 好 迎接 挑战 吧 ! 依赖 注 人 ( Dependency 
Injection )、 现 代 并 发 ( Modern Concurrency ) 和 基于 Java 的 软件 系统 性 能 调 优 ， 这 些 都 等 着 你 去 
探索 ， 把 你 喜爱 的 公 员 ( Duke ) “ 杯 加 满 咖 啡 ， 准备 好 向 前 冲 吧 ! 


了 Duke 是 Java 的 吉祥 物 ! http://kenai.com/projects/duke/pages/Home。 
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本 书 的 这 一 部 分 (第 3 章 ~ 第 6 章 ) 全 部 是 对 Java 中 的 关键 编程 知识 和 技术 的 探索 。 

本 部 分 的 开篇 之 章 是 关于 依赖 和 广 人 的， 这 是 一 项 对 代码 解 粮 并 增强 其 可 副 试 性 和 易 读 性 的 
通用 技术 。 除 了 依赖 注入 的 基础 知识 ， 我 们 还 介绍 了 它 的 演进 过 程 ， 并 探讨 了 一 个 最 佳 实践 是 
如 何 变 成 设计 模式 并 形成 一 个 框架 的 (最 终 其 至 变 成 了 Java 标准 )。 

之 后 ， 我 们 会 探究 出 现在 硬件 领域 的 多 核 CPU 单 命 。 优 秀 的 Java 开发 人 员 要 了 解 Java 的 并 
发 能 力 ， 并 知道 如 何 利 用 它们 充分 发 挥 现代 处 理 器 的 效用 。 尽 管 Java 自 2006 年 (Java 5) 就 大 力 
支持 并 发 编程 ， 但 人 们 对 这 一 领域 的 理解 和 应 用 仍然 很 少 ， 所 以 我 们 会 用 一 整 章 的 内 容 介 绍 它 。 

你 将 看 到 Java 内 存 模型 ， 以 及 这 个 模型 中 的 线程 和 并 发 是 如 何 实现 的 。 一 旦 你 掌握 了 这 些 
理论 知识 ， 我 们 就 会 指 于 你 用 java.util.concurrent 包 及 其 他 一 些 特性 为 Java 并 发 实战 打 
下 基础 。 

接 下 来 ， 我 们 会 介绍 类 加 载 。 很 多 Java 开发 人 员 不 太 明 白 JVM 如 何 加 载 、 链 接 和 验证 娄 。 
所 以 当 革 些 类 的 “错误 ”版 本 由 于 某 种 类 加 载 冲 突 被 执行 时 , 他 们 会 备 感 沪 丧 并 浪费 很 多 时 间 。 

我 们 还 会 谈 到 Java 7 的 MethodHandle、MethodType 和 动态 调用 ， 计 用 Reflection (反射 ) 
编码 的 开发 人 员 能 以 一 种 更 快 、 更 安全 的 方式 完成 相同 的 任务 。 

能 够 保 人 到 Java 类 文件 的 内 部 和 和 它 所 包含 的 字 书 码 中 是 非常 强 的 调试 技能 。 我 们 会 向 你 展 
示 如 何 用 javap 辛 览 和 理解 字 忆 码 的 含义 。 

性 能 调 优 经 常 被 当做 一 门 艺术 ， 而 不 是 科学 。 跟 踪 和 解决 性 能 问题 经 常会 占用 开发 团队 大 
量 的 时 间 和 精力。 在 第 6 章 ， 也 就 是 本 部 分 的 最 后 一 童 ， 我 们 会 教 你 评 囊 (而 不 是 猜测 )， 并 且 
告诉 你 “传说 中 的 调 优 ”是 错误 的 .我们 会 给 你 指出 一 条 直 指 性 能 问题 核心 的 科学 之 路 。 

我 们 特别 关注 垃圾 回收 ‘GC) 和 即时 (JIT) 编译 器 ， 这 是 JVM 中 能 够 影响 性 能 的 两 个 主 
要 部 分 。 除 了 其 他 与 性 能 有 关 的 知识 ， 你 还 将 学 到 如 何 阁 读 GC 日 志 ， 以 及 如 何 用 免费 的 Java 
VisualVM (jvisualvm) 工具 分 析 内 存 的 使 用 情况 。 

读 完 第 二 部 分 之 后 ， 你 就 不 再 是 个 只 想 着 IDE 中 那些 源码 的 开发 人 员 了 。 你 将 知道 Java 和 
JVM 的 内 部 工作 机 制 ,并 能 铝 充 分 发 挥 这 个 星球 上 最 强大 的 通用 VM (这 么 说 并 不 为 过 ) 的 潜力 。 
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本 章 内 容 

口 控制 反 转 (IoC ) 和 依赖 注入 (DI) 

口 掌握 依赖 注 人 技术 为 什么 如 此 重要 

口 JSR-330 如 何 统 一 了 Java 中 的 DI 

口 常见 的 JSR-330 注 解 ， 比 如 @Inject 

口 Guice 3 简介 ，JSR-330 的 参考 实现 ( RI) 


大 约 人 内 2004 年 开始 ,依赖 注 人 ( 控制 反 转 的 一 种 形式 ) 就 是 Java 开 发 主流 中 一 个 重要 的 编程 
范式 ?"。 简 言 之 ， 使 用 Di 技术 可 以 让 对 象 从 别处 得 到 依赖 项 ， 而 不 是 由 它 自 己 来 构造 。 使 用 DI 有 
很 和 多 好 处 ， 它 能 降低 代码 之 间 的 耦合 度 ， 让 代码 更 易于 测试 、 更 易 读 。 

本 章 会 先 对 DI 理论 以 及 其 给 代码 带 来 的 好 处 进行 强化 。 即 便 你 用 过 IoC/DI 框 架 ， 本章 内 容 亦 
能 帮 你 更 深入 地 了 解 DI 的 本 质 。 如 果 你 刚刚 开始 接触 DI 框架 (许多 人 都 是 如 此 )， 那 本 章 中 的 内 
容 对 你 就 尤为 重要 了 。 

你 将 会 了 解 Java DI 的 官方 标准 JSR-330, 并 从 中 了 解 到 Java DI 标准 注解 集 的 幕后 故事 。 随 后 ， 
我 们 会 介绍 JSR-330 的 参考 实现 (RI) Guice 3 一 一 一 个 众所周知 的 轻 量 、 精 巧 的 DI 框 架 。 

我 们 先 来 看 一 些 理论 知识 , 好 让 你 明白 这 个 范式 大 行 其 道 的 原因 , 以 及 你 为 什么 需要 掌握 它 。 


3.1 知识 注入 : 理解 loC 和 DI 


为 什么 需要 了 解 控 制 反 转 ( IoC、 依 赖 注入 ( DI) 以 及 它们 的 基本 原理 ?对 于 这 个 问题 , 仁 
者 见 仁 智 者 见 智 。 如 果 你 在 知名 的 问答 网 站 programmers.stackexchange.com 上 问 这 个 问题 ， 肯 定 
会 得 到 很 多 不 同 的 答案 ! 

你 可 能 只 是 刚 开 始 使 用 不 同 的 DI 框架 并 学 习 网 上 的 示例 ， 但 如 果 你 能 够 掌 所 对象 关系 映射 


( Object Relational Mapping，ORM ) 框架 ， 比 如 Hibemate， 你 就 可 以 变 成 编程 高 手 。 


山 范式 ( paradigm ) 在 1960 年 之 后 是 指 在 科学 领域 和 知识 论 行文 中 的 思维 方式 。 一 一 译 者 注 
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本 节 首 先 介绍 核心 术语 IoC 和 DI 背后 的 一 些 原理 ， 并 探讨 使 用 这 一 范式 的 好 处 。 为 了 让 这 些 
概念 不 至 于 那么 抽象 ,我 们 会 以 HollywoodService 为 例 展示 它 的 转变 过 程 一 一 从 自己 构造 依赖 
项 变 成 被 注入 依赖 项 。 

我 们 先 从 IoC 开 始 ， 这 个 本 请 经 常 被 ( 错误 地 ) 和 DI 互 换 使 用 。* 


3.1.1 控制 反 转 


在 使 用 非 JoC 范 式 编程 时 , 程序 巡 辑 的 流程 通 贡 是 由 一 个 功能 中 心 来 控制 的 。 如 果 设 计 得 好 ， 
这 个 功能 中 心 会 调用 各 个 可 重用 对 象 中 的 方法 执行 特定 的 功能 。 

使 用 IoC， 这 个 “中 心 控 制 ” 的 设计 原则 会 被 反 转 过 来 。 调 用 者 的 代码 处 理 程序 的 执行 硕 施 ， 
而 程序 逻辑 则 筱 封闭 在 接 党 再 用 的 于 流程 中 。 

IoC 也 被 称 为 好 莱 坞 原则 ,其 思想 可 以 归结 为 会 有 男 一 段 代码 拥有 最 初 的 控制 线程 ,并 且 由 
它 来 调用 你 的 代码 ， 而 不 是 由 你 的 代码 调用 它 。 


““ 姑 关 经纪 人 总 是 给 人 条 电 话 ， 而 不 让 到 人 和 打 给 他 们 1 如 果 你 曾经 了 好 匡 志 经 包 人 所 
在 明年 夏天 筹划 一 个 “让 Java 程 序 员 成 为 拯 教 世界 的 英雄 ”的 大 片 ， 你 也 许 会 深 请 其 道 。 


换 一 种 方式 来 看 JoC， 回 想 一 下 视频 游戏 Zork(http://en.wikipedia.org/wiki/Zork) 用 户 界面 的 发 
展 过 程 ， 从 早期 由 命令 行 中 的 文本 控制 到 如 今 用 图 形 用 户 界 面 控 制 。 

在 命令 行 版 本 中 ， 用 户 界面 上 只 有 一 个 空白 提示 符 , 等 着 用 户 输入 。 当 用 户 输入 “向 东 ” 或 者 
“Grue， 快 逃 ” 的 指令 后 ， 游 戏 的 主 应 用 逻辑 会 调用 恰当 的 事件 处 理 器 来 处 理 这 些 指令 ， 并 返回 
结果 。 这 里 的 关键 点 是 应 用 逻辑 要 控制 调用 哪个 事件 处 理 咒 

而 在 GUI 版 本 中 , IoC 开 始 发 挥 作用 。 由 GUI 框架 来 控制 调用 事件 处 理 嚣 , 而 不 是 由 应 用 逻辑 。 
当 用 户 点 击 了 一 个 动作 ， 比 如 “ 癌 东 ”时 ，GUI 框 染 会 且 接 调用 对 应 的 事件 处 理 右 ， 而 应 用 妙 辑 
可 以 把 重点 放 在 处 理 动 作 上 。 

程序 的 主 控 被 反 转 了 ， 将 控制 权 从 应 用 逻辑 中 转移 到 GUI 框 架 。? 

IoC 有 几 种 不 同 的 实现 ,包括 工厂 模式 、 服 务 定位 器 模式 ， 当 然 ， 还 有 依赖 注入 。 这 一 术语 
最 初 由 Martin Fowler 在 “控制 反 转 容器 和 依赖 注 人 模式 ”= 中 提出 ， 然 后 迅速 传 般 大 街 小 埠 ， 反 
响 强 烈 。 


总 


中 从 字面 上 来 看 ，loC 是 指 一 种 机 制 ， 使 用 这 种 机 制 的 用 例 很 多 ， 实 现 方式 也 很 多 。DI 只 是 其 中 一 种 具体 用 例 的 具 
体 实 现 方式 。 但 因为 DI 非常 流行 ， 所 以 人 们 经 常 误 以 为 IoC 就 是 DI， 并且 认为 DI 这 种 叫 法 比 IoC 更 贴切 。 这 是 来 自 
stackoverflow 的 更 全 面 解释 (英文 )，http://stackoverflow.com/questions/6550700/inversion-of-control-vs-dependency- 
injection。 一 -一 详 者 注 

思 程序 中 出 现 了 专门 用 来 实现 调用 和 控制 超 辑 的 GUI 框 架 ， 应 用 逻辑 中 的 代码 只 需 关注 应 用 请 求 的 处 理 。 一 一 译 者 注 

四 在 Martin Fowler 的 网 站 http://martinfowler.com/ 中 搜索 Dependency Imjection， 你 就 可 以 找到 这 箱 文 章 。 
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3.1.2 ”依赖 注入 

依赖 注入 是 loC 的 一 种 特定 形态 , 是 指 寻 找 依赖 项 的 过 程 不 在 当前 执行 代码 的 直接 控制 之 下 。 
虽然 你 也 可 以 自己 编写 代码 实现 依赖 注 人 机 制 ， 但 大 多 数 开发 人 员 都 会 使 用 自 带 IoC 容 器 的 第 三 
方 DI 框架 ， 比 如 Guice。 


注意 可 以 把 loC 容 器 看 做 运行 时 环境 。Java 中 为 依赖 注入 提供 的 容器 有 Guice、Spring 和 


PicoContamer, 


IoC 容 器 可 以 提供 实用 的 服务 ， 比 如 确保 一 个 可 重用 的 依赖 项 会 被 配置 成 单 例 模 式 。 我 们 在 
3.3 节 介绍 Guice 时 会 探讨 它 的 一 些 服务 。 


提示 “把 依赖 项 注入 对 象 的 方法 有 很 多 种 。 可 以 用 专门 的 DI 框 架 ， 但 也 可 以 不 这 么 做 ! 显 式 地 
创建 对 象 实例 (依赖 项 ) 并 把 它们 传 入 对 象 中 也 可 以 和 框架 注入 做 的 一 样 好 。” 


与 很 多 编程 范式 一 样 ， 理 解 使 用 DI 的 原因 很 重要 。 我 们 在 表 3-1 中 总 结 了 它 的 主要 好 处 。 
表 3-1 DI 的 好 处 


好 处 
松 楂 合 


易 娟 性 
更 强 的 内 聚 性 
可 重用 组 件 


更 轻 向 的 代码 


把 普通 代码 改写 成 依赖 注入 的 代码 是 掌握 这 些 


描述 


你 的 代码 不 再 紧 紧 地 畴 定 到 依赖 项 的 创建 上 
了 。 如 果 能 与 面向 接口 编程 的 技术 相 结合 ， 意 
味 着 你 的 代码 再 也 不 用 紧 紧 地 绑 定 到 依赖 项 的 
具体 实现 上 了 

作为 松 粳 合 的 延伸 ， 还 有 个 特殊 的 用 例 值得 一 
提 。 为 了 测试 ， 可 以 把 测试 灶 身 "作为 依赖 项 注 
入 到 对 象 中 

你 的 代码 可 以 专注 于 执行 自己 的 任务 ， 不 用 为 
了 载 人 和 配置 依赖 项 而 分 心 。 这 样 还 能 增强 代 
玛 的 可 读 性 

作为 松 耦 合 的 延伸 ， 你 的 代码 应 用 范围 会 更 加 
宽广 ， 那 些 可 以 提供 自己 特定 实现 的 用 户 都 可 
以 使 用 你 的 代码 

你 的 代码 不 再 需要 跨 层 传递 依赖 项 ， 而 是 可 以 
在 需要 依赖 项 的 地 方 直接 注入 


例 于 
HollywoodService 对 象 趟 再 需要 创建 它 所 需 的 
部 传人 的 对 象 。 如 果 而 向 接口 编程 , Hollywood- 
Service 可 以 楼 受 任 何 类 型 的 AgentFinder 传 人 
你 可 以 注入 一 个 总 是 返回 相同 价格 的 虚设 票 价 
服务 ， 而 不 是 使 用 “真实 ”的 价格 服务 ， 因 为 
入 是 外 衣 服 务 ， 而 且 有 了 时 无 法 访问 
你 的 DAO 对 象 可 以 专注 于 查询 工作 ， 不 用 考虑 
JDBC 驱 动 的 细节 


一 个 积极 进取 的 软件 开发 人 员 可 能 会 卖 给 你 一 
个 Linkedin 代 理 人 查找 六 


你 不 再 需要 把 JDBC 驱 动 的 细 池 信息 从 service 类 
往 下 传递 ， 而 是 直接 在 真正 需要 它 的 DAO 中 直 
接 注入 这 个 驱动 实例 


理论 的 最 佳 方法 ， 所 以 我 们 进入 下 一 节 吧 。 


感谢 Thiago Arrais ( http://stackoverflow.com/users/17301/thiago-arrais ) 提供 了 这 个 提示 。 
四 第 11 章 会 详细 介绍 测试 替身 。 
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3.1.3 转 成 DI 


本 节 会 重点 介绍 如 何 把 不 用 IoC 的 代码 变 成 使 用 工厂 (或 服务 定位 器 ) 模式 的 代码 ， 再 变 成 
使 用 DI 的 代码 。 在 这 些 转 变 之 后 有 一 个 共同 的 关键 技术 ， 即 面向 接口 编程 。 使 用 面向 接口 编程 ， 
甚至 可 以 在 运行 时 更 换 对 象 。 


注意 本 节 的 目的 是 现 因 你 对 DI 的 理解 。 因 此 某 些 比较 套路 化 的 代码 被 当 略 掉 了 。 


假设 你 刚 接 手 了 一 个 小 项 目 ， 要 找 出 所 有 对 Java 开 发 人 员 比 较 友 善 的 好 菜 坞 经 纪 人 。 在 下 面 
的 代码 中 ，AgentFinder 接 口 有 两 个 实现 类 ， spreadsheetAgentFinder 和 WebService- 
AgentFinder. 


代码 清单 3-1 接口 RgentFinder 及 其 实现 类 
Eublic interface MgentFinder 


| 


public List<hgent> findAllhgents!):; 


public class SpreadsheetAgentFinder implements AgentFinder 
{ 

吕 吕 VerTT1ide 

public Liast<Agent> findAllAgenta(){ ... } 


ublic class WebServiceAgentFinder implements AgentFinder aa 
p 9 ? 9 很 多 实现 细节 
EOverrlide 


publiec List<Agents findAllAgents(})}{! ... } 
} 
为 了 使 用 经 纪 人 查找 帮 ， 项 目 中 有 个 默认 的 Hollywoodservic 类 ， 它 会 人 人 spreadsheet- 
AgentFinder 里 得 到 一 个 经 纪 人 列表 4 并 根据 是 理 友 善 对 他 们 进行 过 滤 | 最 终 返 回 友 善 的 经 纪 
人 列表 。 如 下 面 的 代码 所 示 。 


代码 清单 3-2 HollywoodService 一 一 自己 创建 


public class HollywoodService 


gentFinder 的 具体 实现 类 实例 
种 使 用 spreadsheet- 
"AgentFinder 
public static List<Agent> getFriendlyAgents!) 


AgentFinder finder = new SpreadsheetAgentFinder'(}); 二}— | 调用 接口 方法 
| I ld 


List<AMgent> agents = finder.findAllAMgents!(),; 
List<Agent> friendlyAgents = 
filterAgentslagenta, ‘Java Developers").; 


return friendlyAgentes; 
| 返回 友善 的 经 纪 人 
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public static List<Agent> filterAgents (List<Agent> agentas, 


String agentType) 
| 
List<AMgent> filteredhgents = new ArrayLiast<>(); 
for (Agent agent:agents) | 
if (lagent .getType() .equalsl"Java Developers")) | 
filteredAgents.addlagent);} 
| 
} 
return filteredhgents,:; 
} 
} 


下 看 一 遍 上 面 代 码 里 的 Hollywoodservice， 注 意 到 了 吗 ” 它 被 spreadsheetAgentFinder 


这 个 agentFinder 的 具体 实现 死 死 地 夭 上 TO. 


过 去 这 种 黏 糊糊 的 实现 对 Java 开 发 者 来 说 司空 见 惯 ,不 胜 其 烦 ! 为 了 解决 这 些 共性 问题 ， 设 
计 模 式 应 运 而 生 。 一 开始 , 很 多 开发 人 员 用 工厂 模式 和 服务 定位 吾 模 式 的 各 种 变 体 来 解决 这 类 问 


它们 全 都 是 IoC 的 一 种 。 
1. 使 用 工厂 和 /或 服务 定位 器 模式 的 HollywoodService 


抽象 工矿、 工厂 方法 或 服务 定位 器 模式 中 的 某 个 【或 它们 的 某 种 组 合 ) 通常 用 来 解决 这 种 被 


依赖 项 昔 上 的 问题 。 


注意 工厂 方法 和 抽象 工厂 模式 在 Erich Gamma、Richard Helm、Ralph Johnson 和 John Vlissides 
写 的 《设计 模式 可 复 用 面向 对 人 象 软件 的 基础 》( Addison-Wesley Professional， 
有 了 所 论述 。 服 务 定位 器 模式 出 现在 Deepak Alur、John Crupi 和 Dan Malks 编 写 的 Can 


心 模式 》 第 二 版 ( Prentice Hall，2003 ) 中 。 


下 而 这 个 版 本 的 HollywoodService 类 用 AgentFinderFactory 实 现 对 AgentFinder 的 动 


态 选择 。 
代码 清单 3-3 ”Hollywoodservice 由 工厂 负责 提供 agentFinder 


public class HollywoodServiceWithFactory | 


public List<Agent> 
getFriendlyAMAgents (String agentFinderType) 
人 


AgentFinderFactory factory = 

hgentFinderFactory .getlnstancel); 
AgentFinder finder = 

factory .gethgentFinder (agentFinderType):} 
List<Agent> agents = finder.findAllAgents!(); 
List<Agent> friendlyAgents = 

filterAgente (agents, ‘Java Developers'"), 

return friendlyAgentes:; 


只 自 在 读书 @3 


WWwwW .Zizidiary .com 


号 


全 注入 agentFinderType 


通过 工厂 实例 获取 


AgaentFinder 


1994 ) 中 
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public static List<Agent> filterAgents (List<Agent> agents, 
string agentType) 
| 


, 人 ee 与 代码 清单 3-2 中 
”的 实现 一 样 


如 你 所 见 ， 现 在 没有 特定 的 agentFindaer 实 现 类 来 籍 你 了 。 注 和 人 agentFinderType 吉 ,让 
AgentFinderFactory 根 据 这 一 类 型 挑选 令 人 满意 的 AgentFinder 供 你 享用 人 @， 
这 和 DI 相当 接近 了 ， 但 还 有 两 个 问题 。 
口 代码 中 注 人 的 是 一 个 引用 和 凭据 (agentFinderType 上 而 不 是 真正 实现 AgentFinder 的 
对 象 。 
口 方法 getFriendlyagents 中 还 有 获取 其 依赖 项 的 代码 ， 达 不 到 只 关注 目 身 职能 的 理想 
状态 。 
随 着 开发 人 员 编 写 更 清晰 代码 的 意愿 不 断 增 强 , DI 技术 也 越 来 越 流 行 , 正 在 逐步 取代 工厂 和 
服务 定位 器 模式 。 
2. 使 用 DI 的 HollywoodService 
你 可 能 已 经 猜 出 接 下 来 重 构 代 码 该 做 什么 了 ! 下 一 步 要 让 agentFindaer 直 接 提供 所 需 的 
getFriendlyAgents 方 法 。 代 码 如 下 所 示 ; 


代码 清单 3-4 ”Hollywoodservice 一 一 纯 手工 DI 注 人 AgentFinder 
public class HollywoodServiceWithDI 


| 


public static List<Agent> 0 | 注入 AgentPFinder 
emailFriendlyAgents'(lAgentFinder finder) \ : 
| 
ListcAgent>s agents = finder.findAllAgents()}:; 2 
List<Agent> friendlyAgents = 0 执行 查找 逻辑 
filterhgents(lagents, "Java Develcopers"):; 


return friendlyAgents; 


public static List<Agent> filterAgents (List<Agent> agents, 
String agentType) 
I 


3 参见 代码 清单 3.2 


| 
看 看 这 个 纯 手工 打造 的 DI 方案 ，AgentFinder 被 直接 注 人 到 getFriendlyAgents 方 法 中 
各 。 现 在 这 个 getFriendlyaAgents 方 法 干净 利落 ， 只 专注 于 纯 业 务 逻 辑 和 @。 

不 过 这 种 手工 打造 DI 的 生产 方式 还 是 有 让 人 头疼 的 地 方 。 如 何 配 置 AgentFinder 具 体 实现 
的 问题 并 没有 人 解决 ， 原 本 AgentFinderFactory 要 做 的 工作 还 是 要 找 个 地 方 完成 。 

所 以 ， 能 够 真正 让 我 们 脱离 若海 的 只 有 自 带 IoC 容 器 的 DI 框 架 。 打 个 比方 ，DI 框 架 就 是 把 你 
的 代码 包 起 来 的 运行 时 环境 ， 在 你 需要 时 为 你 注入 依赖 项 。 
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DI 框架 的 优势 在 于 它 可 以 随时 随地 为 你 的 代码 提供 依赖 项 。 因 为 框架 中 有 IoC 容 器 ， 在 运行 

时 ， 你 的 代码 需要 的 所 有 依赖 项 都 会 在 那里 准备 好 。 

如 果 Hollywoodservice 类 使 用 标准 的 JSR-330 注 解 ( 可 以 使 用 任何 与 JSR-330 若 容 的 框架 )， 
那么 它 会 是 什么 样子 ? 

3. 使 用 JSR-330 DI 的 HollywoodService 

来 看 看 这 个 例子 的 最 终 版 本 ,用 框架 来 执行 DI 操作 ,在 这 里 ,DI 框架 用 标准 的 JSR-330@Inject 
注解 将 依 藉 项 直接 注 人 到 getFriendlyagents 方 法 中 ， 代 码 如 下 所 示 : 


gentFinder 


代码 清单 3-5 Hollywoodservice 一 一 用 JSR-330 DI 广 人 :; : 

public class HollywoodServiceJjSR330 JSR-330 注 入 

| AgantFinder 
WInject public void emailFriendlyAgentsas (AgentFinder finder) < 


List<Agent> agents = this,finder.findAllAgentsl).; 
List<Agents friendlyAgents = 执行 查找 运 辑 
filterhgents(lagents, "Java Developers"):;} 
return friendlyAgentse,; 
| 
Public static List<Agent> filterAgents lLiast<Agent> agents, 
String agentType) 
| 
} 
} 
现在 AgentFinder 的 某 个 具体 实现 ( 比如 webserviceagentFinder ) 类 的 实例 是 由 支持 
JSR-330eInject 注 释 的 DI 框架 在 运行 时 注 人 的 @， 


: | 参见 代码 清单 3-2 


提示 “尽管 JSR-330 注 解 可 以 在 方法 上 注入 依赖 项 ,但 通常 只 用 于 构造 方法 或 set 方 法 中 。 下 一 节 
会 对 这 一 规范 做 进一步 探讨 。 


让 我 们 结合 代码 清单 3-5 中 的 HollywoodServiceJSR330 类 再 来 重 温 一 下 依赖 注 人 对 我 们 
的 帮助 。 

口 松 艳 合 HollywoodService 不 再 依 环 于 AgentFinder 的 具体 类 来 完成 工作 。 

口 可 测 性 一 一 为 了 测试 HollywoodService 类 你 可 以 注入 一 个 返回 周 定数 量 经 纪 人 的 基 
本 Java 类 ( 比如 POJOAgentFinder )， 这 在 测试 驱动 的 开发 中 被 称 为 存根 类 ,。 这 对 于 单元 
测试 来 说 非常 完美 ， 因 为 你 不 表 需 要 Web 服 务 、 电 子 表 格 或 其 他 第 三 方 实现 之 类 的 东西 。 

口 吊 强 的 内 售 性 一 一 你 的 代码 不 用 再 和 工厂 打交道 ， 不 用 四 下 里 去 抓 依赖 项 ， 只 需要 执行 
业务 逻辑 。 

口 可 重用 的 组 件 一 一 假如 有 个 开发 人 员 想 用 你 的 API， 现 在 需要 注入 一 个 他 们 定制 的 
AgentFinder 实 现 类 ， 就 说 JDBCAgentFinder 吧 ， 相 象 一 下 他 轻松 异 意 的 表情 吧 。 
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口 更 轻 租 的 代码 HollywoodServiceJSR330 中 的 代码 量 与 最 初 的 HollywoodService 
相 比 明显 减少 了 很 多 。 
使 用 DI 正 在 逐步 成 为 优秀 Java 开 发 人 员 的 标准 实践 ， 几 个 流行 的 容 姻 都 提供 了 优异 的 DI 能 
力 。 然 而 就 在 不 久之 前 ，DI 框 架 领域 还 是 群雄 割据 ， 它 们 风格 钵 异 ， 各 行 其 是 ， 使 用 loC 容 冀 的 
配置 标准 都 自 成 体系 。 即 便 遵 循 类 似 配 置 风格 的 框架 ( 比如 XML 或 Java 注 解 )， 还 是 存在 什么 是 
共通 的 注解 和 配置 这 个 问题 。 
新 的 DI 标准 化 方式 (JSR-330 ) 就 是 要 解决 这 个 问题 。 它 对 大 多 数 Java DI 框架 的 核心 能 力 做 
了 很 好 的 汇总 。 因 为 有 DI 框架 ( 比如 Guice ) 的 内 部 工作 机 制作 为 其 坚实 的 基础 ， 所 以 接 下 来 我 
们 会 深入 探讨 这 一 标准 化 方式 。 


3.2 ”Java 中 标准 化 的 DI 


从 2004 年 开始 ， 有 几 个 用 于 依 丈 注入 的 loC 容 器 得 到 了 广泛 的 应 用 ( 仅 以 Spring、Guice 和 
PicoContainer 为 例 )。 曾 几何 时 ， 它 们 在 DI 的 配置 方式 上 仍然 各 自 为 政 ， 这 使 开发 人 员 很 难 在 不 
同 框架 之 间 迁 移 。 

这 一 问题 直到 2009 年 5 月 才 出 现 转机 ，DI 社 区 的 两 大 巨头 Bob Lee ( Guice ) 和 Rod Johnson 

ingSourc 0 袜 布 要 齐心 协力 ， A ee 他 要 了 JSR-330 


ey rp py a 
中 的 上 下 文 及 依赖 注入 ) 规范 确定 ,你 可 在 http://jcp.org/ 中 搜索 JSR-299 了 和 解 其 详细 信息 。 简 言 
之 ，JSR-299 构 建 在 JSR-330 基 础 之 上 ， 和 旨 在 为 企业 应 用 提供 标准 化 的 配置 。2 


自从 javax .inject 出 现在 Java 中 ( Java SE 5、6 和 7 都 支持 ) 以 来 ， 代码 中 就 可 以 使 用 标准 
的 依赖 注入 了 ， 也 可 以 在 不 同 的 DI 框架 中 进行 迁移 。 比 如 ， 你 原来 在 轻 量 级 的 Guice 框 架 中 运行 
的 代码 ， 为 了 利用 其 丰富 的 特性 ， 也 可 以 迁移 到 Spring 框架 中 去 。 


警告 ”实际 上 ,代码 迁移 并 不 容易 。 一 旦 你 的 代码 用 到 了 仅 由 特定 ID 框架 支持 的 特性 ， 就 不 太 可 
能 摆脱 这 一 框 各 了。 尽管 javax.inject 包 提供 了 常用 DI 功能 的 子 集 , 但 是 你 可 能 需要 使 
用 更 珊 级 的 DI 特 性 。 正 如 你 想象 的 那样 ,对 于 哪些 特性 应 该 作为 通用 的 标准 也 是 众说 纷 纸 ， 
很 难 统一 。 虽 然 现 状 不 尽 如 人 意 ， 但 Java 毕 竟 朝 DI 框 架 的 标准 化 方向 迈 出 了 一 步 。 


山 Bob Lee, “Announcing (Mjavax.inject. Inject” ( 2009-05-08 ), www.theserverside.com/news/thread tss?thread jid=54499。 

四 JSR-299 ( Java Contexts and Dependency Injection ) 目前 由 Redhat 的 Gavin King ( Hibermate 的 创建 者 ) 主导 .因为 它 
比较 新 ， 所 以 设计 理 访 上 解决 了 以 前 DI 框架 中 的 一 些 问 题 ， 而 且 也 不 是 非得 在 Java EE 容器 上 才能 使 用 ， 在 Serviet 
容器 上 也 可 以 使 用 。 其 参考 实现 为 weld， 详 情 请 参见 官网 ，http://www.seamframework.org/Weld。 一 一 译 者 注 
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为 了 理解 最 新 的 DI 框架 如 何 应 用 新 标准 , 我 们 需要 对 javax .inject 包 进行 一 看 研究 。 记 住 ， 
javax.inject 包 只 是 提供 了 一 个 接口 和 几 个 注解 类 型 ， 这 些 都 会 被 遵循 JSR-330 标 准 的 各 种 DI 
框架 实现 。 也 就 是 说 ， 除 非 你 在 创建 与 JSR-330 兼 容 的 IoC 容 器 ( 如 果 如 此 ， 向 你 致敬 )， 通 常 不 
用 自己 实现 它们 。 


2 a a 


5 不 能 满足 于 只 文体 类 pp 还 要 明 白 其 内 pp 
原理 ,在 DI 领域 ,不 理解 其 原理 可 能 会 面临 各 种 难 纺 的 问题 ， 比 如 依赖 项 配置 错误 、 依 赖 项 诡 
异地 超出 作用 域 、 依 赖 项 在 不 该 共享 时 被 共享 以 及 分 步调 试 离奇 宕 机 等 。 


javax .inject 的 文档 对 这 个 包 的 目的 做 出 了 精彩 的 解释 ， 所 以 我 们 全 盘 照 搬 过 来 了 : 

javax.inject 包 ， 

这 个 和 包 指 明了 获取 对 章 的 一 种 方式 , 与 情 统 的 构造 方法 、 工 厂 模 式 和 服务 定位 器 模式 (比如 
JNDI ) 等 相 比 ,这 种 方式 的 可 重用 性 、 可 测试 性 和 可 维护 性 都 得 到 了 极 大 提升 。 这 种 方式 称 为 依 
赖 注 入 ， 对 于 大 多 数 非 小 型 应 用 程序 都 很 有 帮助 。 

javax .inject 包 里 包括 一 个 Provider<T> 接 口 和 5 个 注解 类 型 ( @Inject、@Qualifier、 
eNamed、escope 和 esingleton )， 后 面 几 节 会 逐一 对 它们 进行 介绍 。 先 从 8@Inject 开 始 。 


3.2.1 @Inject 注解 


8Inject 注 解 可 以 出 现在 三 种 类 成 员 之 前 ,表示 该 成 员 需 要 注入 依赖 项 。 按 运行 时 的 处 理 顺 
序 ， 这 三 种 成 员 燃 型 是 : 

(1) 构造 阁 

(2) 方法 

(3) 属性 

在 攀 造 器 上 使 用 8@Inject 时 ， 其 参数 在 运行 时 由 配置 好 的 IoC 容 兹 提供 。 比 如 在 下 面 的 代码 
中 ， 运 行 时 调用 MurmurMessage 类 的 构造 器 时 ，IoC 容 器 会 注入 其 参数 Header 和 content 对 象 。 


InNjJect public MurmurMessage (Header header, Content content) 


this.header = header:; 
this.content = content.; 
} 
规范 中 规定 向 构 造 顺 注入 的 参数 数量 为 0 或 多 个 ， 所 以 在 不 会 人 参数 的 构造 器 上 使 用 einject 
也 是 合法 的 。 


OD http:/atinjectegooglecode.comuysvn/trunkyjavadoc/javaxyinjectpackage-sumrmary.html。 
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警告 因为 JRE 无 法 决定 构造 器 注入 的 优先 级 ， 所 以 规范 中 规定 类 中 只 能 有 一 个 构造 器 带 
BeInject 注 解 。 


也 可 以 用 einject 注 解 方法 ， 与 构造 器 一 样 ， 运 行 时 可 注 人 参数 的 数量 也 可 以 是 0 或 多 个 。 
但 使 用 参数 注入 的 方法 不 能 声明 为 抽象 方法 ， 也 不 能 声明 其 自身 的 类 型 参数 ”。 下面 这 段 代码 在 
set 方 法 前 使 用 einject， 这 是 注 人 可 选 属性 的 常用 技术 。 


InjJect DubBlic Vvoid SetContent (Content content) 


this.content = content:; 


} 
问 方法 中 注入 参数 的 技术 对 于 服务 类 方法 来 说 非常 有 用 ， 其 所 需 的 资源 可 以 作为 参数 注入 ， 
比如 加 查询 数据 的 服务 方法 中 注入 数据 访问 对 象 (DAO )。 


提示 。 向 构造 器 中 注入 的 通常 是 类 中 必需 的 依赖 项 , 而 对 于 非 必需 的 依赖 项 , 通常 是 在 set 方 法 上 
注入 。 比 如 已 经 给 出 了 默认 值 的 属性 就 是 非 必需 的 依赖 项 。 这 一 最 佳 实践 已 经 成 了 惯例 。 


也 可 以 直接 在 属性 上 注入 人 ( 只 要 它们 不 是 Einal )， 虽 然 这 样 做 简单 直接 ,但 你 最 好 不 要 用 。 
因为 这 样 可 能 会 让 单元 测试 更 加 困难 。 直 接 注 人 的 语法 也 非常 简单 。 
public class MurmurMessenger 


Inject private MurmurMessage mrmurMessage; 


} 

你 可 以 在 Javadoc 中 了 解 更 多 关于 einject 注 解 的 详细 内 容 ， 可 以 在 其 中 找到 哪些 类 型 的 值 
可 以 和 广 人 以 及 如 何 处 理 依 球 循 环 。 

对 于 eInject， 现 在 你 应 该 不 再 感到 陌生 了 。 接 下 来 就 该 了 解 如 何 限定 (进一步 标识 ) 那些 
注入 到 你 的 代码 中 的 对 象 了 。 


3.2.2 eoualifier 注解 


支持 JSR-330 规 范 的 框架 要 用 注解 eouali fier 限 定 (标识 ) 要 注入 的 对 象 。 比 如 在 IoC 容 器 
中 有 两 个 类 型 相同 的 对 象 ， 当 把 它们 注入 到 你 的 代码 中 时 ， 肯 定 要 把 它们 区 别 开 。 图 3-1 解 释 了 


山 即 不 能 使 用 Dracle 网 站 上 的 Java 教 程 ( http://download.oracle.com/javase/tutorial/extra/generics/methods.html) 中 所 讲 的 
证 型 方法 技巧 。 
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类 型 : | MusicGenre _ 


限定 符 ; | Genre .CLASSICAL 


loC 容 器 获取 限定 


: 本 而 疾 - 直 -本 - 量 P| ee Pe eT ] HMusicGenre 
图 3-1 用 8Qualifier 广 解 区 分 同一 类 型 MusicGenre 的 不 同 bean 


如 果 你 用 过 由 框架 实现 的 限定 符 ， 应 该 知道 要 创建 一 个 8Qualifier 实 现 必 须 遵 循 如 下 
规则 。 

口 必须 标记 为 &Qualifier 和 @Retention (RUNTIME) ， 以 确保 该 限定 注解 在 运行 时 一 直 

有 效 。 

通常 还 应 该 加 上 8@Documented 注 解 ， 这 样 该 实现 就 能 加 到 API 的 公共 Javadoc 中 了 。 

口 可 以 有 属性 。 

D @Target 注 解 可 以 限定 其 使 用 范围 ; 比如 将 其 使 用 范围 限制 为 属性 ， 而 不 是 限定 为 属性 

的 默认 值 和 方法 中 的 参数 。 

为 了 让 你 对 上 面 的 规则 有 直观 的 感受 ， 下 面 给 出 一 个 eaeoualifier 实 现 。 某 音乐 库 框 架 中 的 
IoC 容 器 提供 了 一 个 限定 符 @MusicGenre, 开发 人 员 在 创建 MetalRecordAlbumns 类 时 可 以 使 用 
该 限定 符 ， 以 确保 广 人 了 正确 的 Genre。 

re 

BOuUAalifier 


public minterface MusicGenre 
| 
Genre genre(} default Genre,TRRANCE: 
public enum GENRE | CLASSICAL, METAL, ROCK, TRANCE | 


| 


public class MetalRecordAlbumns 


mInject ®@MusicGenre (GENRE.METAL) Genre genre; 
| 
Java 开 发 人 员 一 般 不 需要 自己 创建 eoualifier 注 解 , 但 要 对 各 种 loC 容 器 如 何 实现 限定 有 个 
基本 的 了 解 。 
JSR-330 规 范 中 要 求 所 有 1IoC 容 器 都 要 提供 一 个 默认 的 aoualifier 注 解 : eNamed。 
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3.2.3”@Named 注解 


eNamed 是 一 个 特别 的 @Qualifier 注 解 ， 借 助 @eNamea 可 以 用 名字 标明 要 注入 的 对 象 。 将 
eNamed 和 8@Inject 一 起 使 用 ， 符 合 指定 名 称 并 且 类 型 正确 的 对 象 会 被 注 人 。 
在 下 面 这 个 例子 中 , 注 人 了 名 称 为 “murmur” 和 “broadcast” 的 两 个 MurmurMessage 对 象 。 


Eublic class MurmurMessenger 


| 


EINnject @Named ("murmur'") private MurmurMessage murmurMessage,; 
Inject @Named(l"broadcast") private MirmurMessage broadcastMessage; 
i 
尽管 还 有 其 他 比较 常用 的 限定 注解 , 但 最 终 只 有 eNamed 被 选 作 JSR-330 的 标准 限定 注解 ， 所 
有 DI 框 染 都 要 实现 。 
发 起 规范 的 各 方 支持 者 还 在 男 外 一 个 问题 上 达成 了 一 致 一 一 同意 用 标准 化 接口 来 确定 注入 
对 象 的 生命 周期 。 


3.2.4 @Scope 注解 


esScope 注 解 用 于 定义 注 人 峰 ( 即 IoC 容 器 ) 对 注入 对 象 的 重用 方式 。JSR-330 规 范 中 明确 了 
如 下 几 种 默认 行为 。 

口 如 果 没 有 声明 任何 escope 广 解 接 口 的 实现 ， 注 人 顺应 该 创建 广 人 对 象 并 且 仅 使 用 该 对 象 

一 次 。 
OQ 如 果 声 明了 escope 注 解 接 口 的 实现 ,那么 和 注 人 对 象 的 生命 周期 由 所 声明 的 aeScope 注 解 实 
现 决 定 。 

口 如 果 注 入 对 和 象 在 8@Scope 实 现 中 要 由 多 个 线程 使 用 ， 则 需要 保证 注入 对 象 的 线程 安全 性 。 

关于 线程 及 线程 安全 的 更 多 细节 ， 请 参阅 第 4 章 。 

口 如 果 某 个 类 上 声明 了 多 个 @scope 注 解 ， 或 声明 了 不 受 支 持 的 @Scope 注 解 ，IoC 容 器 应 该 

抛 出 异常 。 

Di 框架 管理 注 人 对 象 的 生命 周期 时 不 会 超出 这 些 默 认 行 为 划 定 的 界限 。 有 些 IoC 容 器 支持 自 
己 特 有 的 escope， 尤 其 是 在 Web 前 端 领域 ， 至 少 在 JSR-299 被 广泛 应 用 之 前 是 这 样 。 因 为 大 家 公 
认 的 通用 ascope 实 现 只 有 esingleton 一 个 ， 所 以 JSR-330 规 范 中 仅 确 定 了 它 这 么 一 个 标准 的 生 
命 周 期 注解 。 


3.2.5 @Singleton 注解 
esingleton 注 解 接口 在 DI 框架 中 应 用 广泛 。 在 需要 注 人 一 个 不 会 改变 的 对 象 时 ， 就 要 用 


@Singleton., 
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单 例 设计 模 Tamma., 

Ralph Johnson， 和 John Vlissides 合 著 的 《设计 模式 : 可 复 用 面向 对 象 软件 的 基础 》 
( Addison-Wesley Professional, 1994 ) 第 127 页 。 请 谨慎 使 用 单 例 模式 ， 因 为 它 有 时 候 会 变 成 反 
模式 。 


大 多 数 DI 框 架 都 将 esingleton 作 为 注 和 人 对象 的 默认 生命 周期 ， 无 需 显 式 声 明 。 也 就 是 说 如 
果 没 有 明确 指定 注入 对 象 的 生命 周期 , 框架 就 会 认为 你 想 注 人 一 个 单 例 对 象 。 如 果 你 想 显 式 声明 
它 为 单 例 对 象 ， 可 以 用 下 面 这 种 方式 : 

public MurmurMessade 

BInject &@Singleton MessageHeader defaultHeader:; 

| 
在 这 个 例子 中 ， 我 们 假定 defaultHeader 从 来 不 会 改变 ( 切实 有 效 的 静态 数值 )， 所 以 它 可 
以 作为 单 例 对 象 注 入 。 

最 后 ， 我 们 来 讨论 一 下 当 你 觉得 标准 注解 无 法 满足 你 的 需求 时 该 怎么 办 。 


3.2.6 接口 erovider<T> 


如 果 你 想 对 由 DI 框 架 注 和 代码 中 的 对 象 拥 有 更 多 的 控制 权 ， 可 以 要 求 DI 框 架 将 
Provider<T> 接 口 实现 注入 对 象 ”(T) 。 控 制 对 象 的 好 处 在 于 : 

口 可 以 获取 该 对 象 的 多 个 实例 。 

口 可 以 延迟 获取 该 对 象 ( 延迟 加 载 )。 

口 可 以 打破 循环 依赖 。 

口 可 以 定义 作用 域 ， 能 在 比 整个 被 加 载 的 应 用 小 的 作用 域 中 查找 对 象 。 

该 接口 仅 有 一 个 T get1() 方 法 ， 这 个 方法 会 返回 一 个 构造 好 的 注 人 对 象 (T) 。 例 如 ， 在 
MurmurMessage 和 需要 依 玩 项 Mes sage 对 象 时 可 以 问 其 构造 方法 中 注 人 对 应 的 Provider<T> 接 
口 实现 的 实例 ( Provider<Message> )。 根 据 限定 条 件 的 不 同 ， 得 到 的 Message 对 象 也 会 不 同 。 
请 看 下 面 的 代码 。 


代码 清单 3-6 ”使 用 接口 Provider<T> 
import com.google.inject .Inject.; 
import com.google. inject .Provider; 


class MurmurMessage 


BInject MurmurMessage (Provider<Message> messageProvider) 
得 到 一 个 Message 


MessacGe magl = messaogqeProvider .getl):; 


山 原文 如 此 ， 应 为 该 类 ， 后 面 还 有 几 处 笔 误 ， 请 注意 。 一 一 详 者 广 
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if (someGlobalcondition) 


MesSsage coOPpyOfMSgl = messageProvider .get(); 才 一 一 一 
| © 得 到 Message 的 
复 本 


|} 
} 


注意 上 面 的 代码 中 是 如 何 从 Provider<Message> 中 获取 第 二 个 Message 对 象 的 。 如 果 直 接 
注 人 Message， 就 无 法 获取 另外 一 个 Message 实 例 。 在 这 个 例子 中 ,第 二 个 注 和 对象 仅 在 需要 时 
才 会 加 载 @， 

我 们 已 经 对 新 javax . inject 包 背后 的 理论 做 了 介绍 ， 还 给 出 了 一 些 例子 进行 讲解 。 现 在 就 
该 挽 起 袖子 ， 用 成 熟 的 DI 框架 Guice 实 际 操练 一 把 了 。 


3.3 Java 中 的 DI 参考 实现 : Guice 3 


Guice ( 读 作 “Juice”) 是 由 Bob Lee 在 大 约 2006 年 发 起 的 开源 项 目 ， 项目 站 点 地 址 为 http:// 
code.google.com/p/google-guice/。 你 可 以 在 网 站 上 看 到 该 项 目的 设计 初 囊 、 相 关 文 档 并 下 载运 行 
本 节 示 例 代码 所 需 的 二 进 制 JAR 文 件 。 

在 Guice 这 样 的 DI 框架 里 ， 你 可 以 配置 依赖 项 、 绑 定 依赖 项 ， 并 在 使 用 @Inject 注 解 ( 和 它 
在 JSR-330 中 的 朋友 们 ) 注入 依赖 项 时 声明 它们 的 作用 域 。 

Guice 3 是 JSR-330 规 范 的 完整 参考 实现 , 本 节 所 有 内 容 都 是 基于 Guice 3 的 。 虽然 Guice 不 仅仅 
是 一 个 简单 的 DI 框架 ,但 我 们 关注 的 主要 还 是 它 的 DI 能 力 和 示例 代码 ， 其 中 你 可 以 使 用 JSR-330 
标准 注解 和 Guice 一 起 编写 DI 代码 。 

3.3.1 Guice 新 手指 南 

现在 你 已 经 了 解 JSR-330 的 各 种 注解 了 了。 可 以 傅 助 Guice 构 建 一 个 Java 广 和 人 对 象 集合 【包括 它 
们 的 依赖 项 )，。 用 Guice 的 话说 ,为 了 让 注入 器 创建 对 象 关系 图 ， 需 要 创建 声明 各 种 绑 定 关系 的 模 
块 ， 其 中 绑 定 是 用 来 明确 要 注 人 的 具体 实现 类 的 。 量 了 设 ? 不 用 担心 ， 看 到 代码 你 就 明白 了 ， 这 
些 概 念 实际 上 非常 简单 。 


提示 对 拿 关 系 图 、 纪 定 、 模 块 和 注入 器 都 是 Guice 里 的 常用 语 ， 如 果 你 想 用 好 Guice， 最 好 尽 
快 搞 清楚 它们 是 什么 意思 。 


本 节 我 们 还 是 用 Hollywoodservice 的 例子 。 一 开始 先 创建 一 个 拥有 各 种 绑 定 关系 的 配置 类 
(模块 )。 实 际 上 这 是 Guice 框 架 要 管理 的 那些 依赖 项 的 外 部 配置 。 
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最 新 版 的 Guice 3 可 hp eode 0 ge I 


en ey 


为 了 在 Java 代 却 中 使 用 Guice， Peony ys ASS 


我 们 先 来 创建 一 个 定义 银 
展 JAbstractModule,， 绑 定 关系 在 重 写 Hconfigure () 方 法 中 声明 ~ 
Hollvywoodserviece 要 求 eInject 一 个 AgentFinder 的 时 候 ， 就 会 绑 定 WebServiceagent- 
Finder 类 作为 注 和 对象 。 我 们 在 这 里 遵 入 


对 于 本 书 中 后 续 代 码 示例 而 言 ， 构 建 Maven 时 也 会 自动 下 载 Gt 


构造 方法 注入 的 惯例 ， 具 体 实现 请 见 代码 ; 


代码 清单 3-7 Hollywoodservice 一 一 用 Gui 


import com.google.inject .AbatractModule:; 


扩展 AbstractModule 


public class MgentFinderModule extends AbatractModule 


| 


| 


public class HollywoodServiceGuice 


| 


BOverride 一 
protected void configure'!l) | 重 写 configure() 方 法 


| 


bindlAgentFinder.class). 
tolWebServiceAgentFinder.claas).:; 
| 


实现 类 
private AgentFinder finder = null.; 
EInject 
public HollywoodServiceGuice (AgentFinder finder) 


| 


thias.finder = finder:; 


】 

public List<Agent> getFfriendlyAgents |) 

1 
LiateAgent> agents = finder.findAllAgents'(}); 
List<hgent> friendlyMgents = filterAgentslagents, "Java Developers"):; 
return triendlyAgents:; 

| 

public List<Agent> filterAgents(List<Agents agents, String agentType) 


人 


} | 同 代码 清单 3-2 


1 入 
三 


是 关系 的 AgentFinderModule。 这 个 AgentFinderModule 类 扩 
在 本 例 中 ， 当 客户 类 


绑 定 要 注入 的 


3.3 Java 中 的 DI 参 者 实现 : Guice3 6] 


定 的 类 ( AgentFinder ) 传 给 它 ， 


绑 定 关系 的 确立 在 调用 Guice 的 bind 方 法 时 发 生 ， 把 要 线 
然后 调用 to 方法 指明 要 注 人 到 哪个 实现 类 @， 

现在 已 经 在 模块 中 声明 了 绑 定 关系 , 可 以 让 注 人 器 构建 对 和 象 关系 图 了 。 接 下 来 我 们 要 看 看 在 
独立 Java 程 序 和 Web 应 用 程序 这 两 种 情况 下 分 别 要 如 何 实现 。 

1. 构建 Guice 对 象 关 系 图 一 一 独立 Java 程 序 

在 标准 的 Java 程 序 中 ， 可 以 通过 publiec static void main{(Sstringl[] args) 方 法 构建 
对 象 关系 图 。 代 码 清单 3-8 如 下 所 示 。 


代码 清单 3-8 HollywoodserviceClient 一 一 用 Gn 
import com.google.inject .Guice; 
import com.google.inject .Injector:; 
import Java.util.List.; 


public class HollywoodServiceClient 
| 
public static void matniScring[] args) 
| 
Injector injector = 
Gulice .createInjector (new MgentFinderModule'\})); 


HollywoodServiceGuice hollywoodService = 
injector .getIinstance (HollywoodServiceGuice.class):; 
List<Agent> agents = hollywoodService.getFriendlyAgentsl):; 


| 
} 
对 于 Web 应 用 ， 人 情况 稍 有 不 同 。 
2. 构建 Guice 对 象 关 系 图 一 一 Web 应 用 程序 
在 Web 应 用 程序 中 ， 需 要 把 guice-servlet.jar 加 到 Web 应 用 的 类 库 中 ， 然 后 在 web.xml 中 添加 下 
面 的 配置 项 : 


filters 
<filter-namesgquiceFilter</filter-names 
filter-class>com.go0gle.inject .servlet .GuiceFiltere/filter-classs 

</filters 

filcrer-mapping> 
<filter-name>guiceFilter</filter-name> 
<Url-patterns/*</url-patterns 

</filter-mapping> 

然后 是 标准 动作 ， 扩 展 servletContextListener 以 便 使 用 Guice 的 ServletModule (与 

代码 清单 3-7 中 的 AbstractModule 类 似 ), 

Public class MyGuiceServletConfig extends GuiceServletContextListener | 

BOverride 


protected Injector getInjector() | 
return Guice.createIniectorinew ServletModule(})):; 


| 
| 
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最 后 一 步 ， 把 下 面 这 些 配置 加 到 web.xml 文 件 中 ， 以 便 servlet 容 器 在 部 署 应 用 时 触发 该 类 。 


<l1i18tener> 
<l]istener-class>com. javaTdeveloper .MyGuiceServletConfig</listener-class> 
</liateners> 


经 由 注 人 带 创 建 Ho1lywoodSserviceGuice, 你 得 到 了 一 个 配置 完备 的 类 , 马上 就 可 以 调用 
其 中 的 getFriendlyAgents 方 法 。 

非常 简单 ， 对 不 对 ? 没 错 ， 但 这 种 把 webserviceAgentFinder 绪 定 到 AgentFinder 上 的 
情况 很 简单 ， 而 你 所 需要 的 绑 定 方式 可 能 要 比 这 个 复杂 ,所 以 我 们 还 需要 了 解 一 下 如 何 定义 更 复 
杂 的 乡 定 方式 。 


3.3.2 水手 强 结 : Guice 的 各 种 绑 定 


Guice 提 供 了 多 种 绑 定 方式 ， 官 方 文 档 列 出 的 绑 定 类 型 如 下 所 示 ， 

口 链接 绑 定 

口 绑 定 注解 

口 实例 绑 定 

口 &Provides 方 法 

口 Provider 绑 征 

D 无 目标 绑 定 

口 内 置 绑 定 

口 即时 绑 定 

我 们 无 意 在 此 重复 Guice 的 官方 文档 ， 因 此 只 挑 最 常用 的 讲解 一 下 ， 其 中 包括 链接 绑 定 、 绑 
定 注解 和 epProvides 方 法 及 Provider<T> 绑 是 。 

1. 链接 绑 定 

链接 绪 定 是 最 简单 的 绑 定 方式 ,代码 清单 3-6 中 配置 AgentFinderModaule 时 用 的 就 是 这 种 方 
式 。 这 种 绑 定 方式 只 是 告诉 注入 六 运行 时 应 该 注 人 实现 类 或 扩展 类 ( 是 的 , 可 以 直接 注入 子 类 )。 

BODVeIT1iGe 


brotected void configure'!) 


bind(lAgentFinder.class) .tolWebServicehgentPFinder .class):; 


你 已 经 多 过 这 种 绑 定 代 介 卫 ， 让 我 们 看 看 另外 一 种 最 常用 的 绑 是 方式 : 绑 定 注解 。 

2. 绑 定 注解 

绑 定 注解 是 指 将 注入 类 的 类 型 和 额外 的 标识 符 组 合 起 来 以 标识 恰当 的 注入 对象。 你 可 以 自 
定义 绑 定 注解 (参见 Guice 在 线 文档 ), 不 过 我 们 还 是 先 介 绍 一 下 JSR-330 标 准 注解 eNamed 的 用 法 。 

在 下 面 的 例子 中 ， 你 所 熟悉 的 einject 依 然 会 出 现 ， 但 它 这 次 会 与 eNamed 注 解 联 被 登场 ， 
以 注 人 特定 名 称 的 agentFinder。 为 了 配置 eNamea 绑 定 ， 需 要 在 AgentModule 中 调用 
annotatedwith 方 法 ， 代 码 清单 3-9 如 下 所 示 : 


1 二 
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代码 清单 3-9 HollywoodService 
PUBl1ic class HollywoodService 
| 


private AgentFinder finder = null; 


号 Inject 
Public HollywoodService I@Named ("primary") AMgentFinder finder) 
this.finder = finder:; 使 用 eNameda 
注解 


| 


public class AgentFinderModule extends AbetractModule 
| 

BOverride 

protected void configure') 


{ 


人 A 
eebservicehgentpinaer el | 生生 

】 

现在 你 已 经 知道 如 何 配置 命名 依赖 项 了 ， 可 以 继续 学 习 另 一 种 绑 定 方式 了 。 下 面 就 用 
epProvides 注 解 和 Providader<T> 接 口 绑 定 完 全 由 你 自己 定制 的 依 理 项 吧 。 

3. @Provides 和 provider: 提供 完全 定制 的 对 象 

你 可 以 用 epProvides 注 解 ， 或 者 在 configuref) 方 法 中 绑 定 ， 以 返回 一 个 完全 由 你 自己 和 填 
制 的 对 象 。 比 如 倪 ， 你 可 能 想 注 人 一 个 非常 特别 的 SpreadsheetaAgentFindaer ( 微软 的 Excel 
电子 表格 实现 )。 

注 人 兹 会 查看 所 有 标记 了 @Provides 注 解 方 法 的 返回 类 型 , 以 决定 要 注 人 哪个 对 象 。 比 如 在 
下 面 的 代码 中 ，HollywoodService 会 用 由 provideAgentFinder() 方 法 提供 的 并 带 有 
eprovidaes 注 解 的 AgentFinder。 


public class AgentFinderModule extends AbstractModule 


{ 


Override 

protected void configure(}( | 返回 注入 器 
甸 Provides 需要 的 类 型 
AgentFinder provideAgentFinder |) 村 = 


| 


spreadsheetAgentFinder finder = 
new SpreadsheetAgentFinder(); 
finder.setType ("Excel 97"),; 
finder.setPathl(l"cC:/temp/agents.xls"):}; 
return finder; 创建 SspreadsheetAgantFinder 
| 的 实例 井 设 定 具体 值 


只 自 在 读书 @3 


WWwwW .Zizidiary .com 
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epProvides 方 法 会 变 得 越 来 越 多 , 为 了 不 把 模块 类 撑 爆 , 你 可 能 要 把 它们 拆 分 出 去 建立 自己 
的 类 。 因 此 ，Guice 支 持 JSR-330 的 Provider<T> 接 口 ; 如 果 你 还 记得 第 3.2.6 节 的 内 容 ， 应 该 不 会 
iam get () 方 法 。 当 在 AgentFinderModule 类 中 遂 过 toProvider 方 法 瑚 定 到 AgentFinder- 
Provider 了 时， 就 会 调用 这 个 方法 。 代 码 如 下 所 示 : 


代码 清单 3-11 AgentFinderModule 一 一 使 用 Provider 接 口 
bublic class AgentFinderProvider implements Provider<AgentFinder> 
BOverride 
public AgentFinder get|() ; , 
z 司 用 m get() 方 法 
SpreadsheetAgentFinder finder = new SpreadsheetAgentFinder(); 
finder.setType ("Excel 97°"); 
finder .setPFath("cC:/temp/agents,xls"); 
return finder:; 
| 
) 


public class MgentFinderModule extends AbastractModule 


| 
BOverride 
protected void configure{) 


| 
bind(lAgentFinder.clagss) 
, i (AgentFinderProvider.clasas):; el Wt 
) 
} 
这 是 最 后 一 个 关于 绑 定 的 例子 。 现 在 你 应 该 能 用 Guice 绑 定 你 需要 的 依赖 项 了 。 但 我 们 还 没 
讨论 依赖 项 的 生命 周期 范围 。 了 解 生 命 周 期 非常 重要 ， 因 为 如 果 对 象 生命 周期 设置 错误 的 话 , 它 
们 可 能 会 存在 更 长 时 间 并 占用 更 多 的 内 存 空 间 。 


3.3.3 在 Guice 中 限定 注入 对 象 的 生命 周期 


Guice 为 注 人 对 象 提供 了 不 同 级 别 的 生命 周期 。 其 中 最 短 的 是 eRequestScope， 然 后 是 
esessionSscope， 还 有 就 是 JSR-330 规 范 中 定义 的 esingleton， 也 就 是 应 用 级 别 的 生命 周期 。 

在 代码 中 有 以 下 几 种 方式 应 用 依赖 项 的 生命 周期 : 

口 在 要 注入 的 类 中 ， 

口 作为 绑 定 声明 的 一 部 分 ( 比如 bina() .to() .in() ); 

口 和 epProvides 一 起 使 用 注解 声明 。 

上 面 的 列表 有 点 儿 抽 象 , 我 们 还 是 用 限定 依赖 项 生命 周期 的 一 小 段 代 码 来 说 明 上 面 这 些 方式 
究竟 是 什么 意思 吧 。 

1. 限定 注入 项 的 生命 周期 

假设 你 希望 程序 的 整个 生命 周期 中 只 用 一 个 spreadsheetAgentFinder 实 例 。 为 此 需 在 类 
声明 中 设置 @singleton， 如 下 所 示 : 
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岛 Singleton 
public class SpreadsheetAgentFinder 


} 

用 这 个 方法 还 有 个 好 处 ， 可 以 提醒 开发 人 员 注 入 项 对 线程 安全 的 要 求 。 因 为 从 理论 上 来 说 ， 
SpreadsheetAgentFinder 类 可 以 多 次 注 人 ，8@singleton 范 围 意 味 着 要 保证 这 个 类 的 线程 安 
全 性 ( 第 4 章 会 讨论 线程 安全 )。 

如 果 你 更 喜欢 在 绑 定 依赖 项 时 声明 其 生命 周期 ， 你 就 可 以 这 样 做 。 

2. 用 BIND (1) 方 法 设置 生命 周期 

有 些 开 发 人 员 可 能 剖 欢 把 所 有 和 广 人 对 得 相关 的 规则 都 放 在 一 起 。 还 记得 代码 清单 3-9 中 如 
何 绑 定 “primary”aAgentFindezr 吗 ? 在 绑 定 时 设置 生命 周期 与 此 类 似 ， 只 要 在 bina1( ) 方 法 串 后 
面 再 加 上 .in(<Scope> .class) 就 行 了 。 

下 面 的 代码 改进 了 代码 清单 3-9， 在 bina() 方 法 串 后 面 加 上 了 in(Session.class)， 使 得 
在 会 话 ( session ) 施 围 中 能 够 获取 “primary”AgentFinder 对 依 。 


public class AgentFinderModule extends AbstractModule 


BOverride 
protected void econfigure!) 
人 
bindlAgentFinder , 它 ] 吉 自白】 
.annotatedWith (Names .named ("primary")) 
.tolWebServiceAgentFinder.class) 
.inlSession.class).; 
| 
} 
还 有 最 后 一 种 设置 注 人 对 和 象 生命 周期 的 方法 : 与 eprovides 注 解 联合 设置 。 
3. 设置 @Provides 对 象 的 生命 周期 
你 可 以 在 8Provides 注 解 旁 边 加 上 一 个 生命 周期 ， 以 定义 由 该 方法 所 提供 对 象 的 生命 周期 。 
比如 在 代码 清单 3-9 中 ， 可 以 加 上 8@Request 注 解 ， 将 最 终 提供 的 spreadsheetAgentFinder 实 
例 限 定 在 请 求 (request ) 范围 中 。 
Provides BRedquest 
AgentFinder provideAgentFinder'|) 
SpreadsheethAgentFinder finder = new SpreadsheethAgentFinder'(); 
finder .setTypel"Excel 97"),; 
finder.setPathl"C:/temp/agents.xls"), 
return finder; 
} 
Guice 也 提供 了 针对 Web 应 用 的 注入 项 生命 周期 ( 比如 Servlet request 范 围 )， 当 然 你 也 可 以 根 
据 需 要 实现 自己 的 注入 项 生命 周期 。 
现在 你 已 经 基本 掌握 了 在 Guice 中 用 JSR-330 注 解 满足 你 的 依赖 注 人 需求 了 。 其 实 Guice 还 提 
供 许 多 非 JSR-330 特 性 ， 比 如 它 还 支持 面向 方面 的 编程 ( AOP )， 让 你 可 以 实现 安全 性 和 日 志 处 理 
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的 横 切 关注 点 。 要 了 解 这 些 内 容 ， 请 参考 Guice 的 在 线 文档 和 代码 示例 。 


i 


re 过 是 会 rep pp pr 
低廉 ,无 需 考虑 其 生命 周期 。JVM 会 在 需要 时 创建 和 销毁 它们 ， 毫 无 障碍 。( 第 6 章 详细 讨论 了 
JVM 和 性 能 。) 

另 一 方面 ,状态 对 象 总 是 需要 设 定 生命 周期 你 应 该 认真 考虑 是 否 要 让 一 个 对 象 的 生命 
期 贯穿 整个 应 用 程序 的 运行 期 或 者 仅 存在 于 当前 会 话 , 还 是 只 在 当前 请 求 中 。 接 下 来 就 要 考 


虑 对 象 的 线程 安全 性 了 ( 第 4 章 会 进一步 讨论 这 个 问题 ). 


3.4 ”小结 


IoC 是 个 复杂 的 概念 。 但 通过 对 工厂 和 服务 定位 器 模式 的 探讨 ,你 能 了 解 基本 IoC 实 现 是 如 何 
工作 的 。 工厂 模式 有 助 于 你 理解 DI 以 及 DI 给 代码 带 来 的 好 处 。 即 便 DI 范 式 接受 起 来 非常 困难 , 它 
也 值得 你 继续 坚持 ， 因 为 它 能 让 你 编写 松散 耦合 的 代码 ， 让 代码 更 易 测 试 和 更 易 读 。 

JSR-330 不 仅仅 是 统一 DI 通 用 功能 的 重要 标准 ， 它 还 提供 了 你 需要 了 解 的 幕后 规则 及 限制 。 
通过 研究 标准 DI 注解 集 ， 你 会 更 加 欣赏 不 同 DI 框 架 对 规范 的 实现 ， 因 而 可 以 更 有 效 地 使 用 它们 。 

Guice 是 JSR-330 的 参考 实现 ， 同 时 它 也 是 一 个 流行 的 、 轻 量 级 的 DI 框架 。 实 际 上 ， 对 于 很 多 
应 用 程序 来 说 ， 使 用 Guice 和 与 JSR-330 兼 容 的 注解 集 可 能 就 足以 满足 你 对 DI 的 需求 了 。 

如 果 你 是 从 头 开 始 看 这 本 书 的 . 我 们 认为 你 应 该 稍 事 休 息 了 ! 放下 手中 的 书 , 去 做 些 别 的 事 
情 ， 然 后 再 精神 饱满 地 回来 继续 阅读 下 一 主题 一 所 有 优秀 的 Java 开 发 人 员 都 应 该 掌握 的 并 发 。 
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本 章 内 容 

口 并 发 理论 

D 块 结构 并 发 

口 java.util.concurrent 包 
口 用 分 支 /合并 框架 实现 轻 量 并 发 
口 Java 内 存 模型 ( JMM ) 


本 章 会 从 基本 概念 开始 ， 并 在 块 结构 并 发 上 做 短暂 停留 。 在 Java 5 之 前 ， 这 是 唯一 值得 把 玩 
的 技术 ,就 是 搁 在 现在 ,也 很 值得 玩味 。 之 后 ,我 们 会 介绍 每 个 开发 人 员 都 应 该 了 解 的 
Java .util .concu rrent 包 以 及 如 何 使 用 其 提供 的 基本 并 发 构建 块 。 

我 们 会 以 新 的 分 支 / 合 并 框架 为 结尾 。 看 完 本 章 后 ， 你 就 有 能 力 使 用 这 些 新 的 并 发 技术 丁 。 
你 还 能 掌握 充分 的 理论 知识 ， 并 以 此 为 基础 ， 能 够 完全 理解 本 书后 面 讨论 到 的 非 Java 语 言 的 不 同 
并 发 视图 。 

我 们 不 打算 在 本 章 把 所 有 的 并 发 知识 都 讲 完 ， 先 告诉 你 一 些 起 步 的 知识 就 足够 了 了 。 万 外 我 们 
还 会 告诉 你 在 编写 并 发 代码 时 需要 深信 了 解 些 什 么 , 并 阻止 你 在 编写 代码 时 涉足 险 境 , 但 如 果 你 
想 成 为 一 名 真正 一 流 的 多 线程 编程 高 手 ， 只 知道 这 里 讲 的 知识 还 不 够 。 这 里 向 你 推荐 两 本 专门 讨 
论 Java 并 发 编程 的 的 书 , 其 中 一 本 是 Doug Lea 写 的 《Java 并 发 编程 》( 第 二 版 , Prentice Hall, 1999 )， 
人 合 着 的 《Java 并 发 编程 实战 》( Addison-Wesley Professional，2006)。 


另 一 本 是 和 Brian GoetzE 


yr yr pe rp 
和 语言 层面 那些 原始 的 基础 Java 并 发 机 制 就 是 合格 的 并 发 编程 人 员 了 。 实 际 上 ， 并 发 是 个 非常 
广泛 的 主题 ， 即 便 是 最 好 的 开发 人 员 ， 从 事 多 线程 开发 工作 多 年 ， 有 直 富 的 经 验 ， 要 起 写 出 估 
质 的 多 线程 代码 也 很 困难 ， 而 且 很 难保 证 不 出 问题 。 

还 有 一 点 休 应 该 注意 目前 并 发 领域 正如 火 如 其 地 开展 着 研究 工作 ,这 些 研究 表 定 会 对 你 
所 用 到 的 Java 及 其 他 语言 产生 影响 。 如 果 非 要 我 们 挑 一 个 在 未 来 五 年 中 很 可 能 会 改变 行业 惯例 
的 计算 机 基础 领域 ， 那 就 是 并 发 。 
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本 章 的 目的 是 让 你 了 解决 定 Java 并 发 工作 方式 的 底层 平台 机 制 。 我 们 还 会 充分 讨论 通行 的 并 
发 理论 和 词汇 , 让 你 理解 其 中 涉及 的 问题 ,并 教 你 认识 到 让 并 发 正确 工作 的 必要 性 和 在 这 个 过 程 
中 所 过 到 的 困难 。 实 际 上 ， 这 里 是 我 们 的 起 点 。 


4.1 并 发 理论 简介 


为 了 理解 在 Java 中 编写 并 发 程序 的 方法 ， 我 们 来 聊 聊 相关 理论 。 先 讨论 一 下 Java 线 程 模型 的 
基础 知识 。 
之 后 ,我们 会 讨论 系统 设计 和 实现 中 “设计 原则 ”的 影响 以 及 其 中 最 主要 的 两 个 原则 : 安全 
性 和 活跃 度 。 我们 还 会 提 到 其 他 一 些 原 则 ,然后 讨论 这 些 原则 经 常 相互 冲突 的 原因 ， 以 及 并 发 系 
统 中 为 什么 会 有 开销 。 
本 节 的 最 后 我 们 会 看 一 个 多 线程 系统 的 例子 ， 并 向 你 证 明 java .util.concurrent 是 多 么 
自然 的 编码 方法 。 
4.1.1 解释 Java 线程 模型 
Java 线 程 模型 建立 在 两 个 基本 概念 之 上 
口 共享 的 、 默 认可 见 的 可 变 状 态 
口 抢占 式 线程 调度 
我 们 从 几 个 侧面 思考 一 下 这 两 个 概念 。 
口 所 有 线程 可 以 很 容易 地 共享 同一 进程 中 的 对 象 。 
口 能 够 引用 这 些 对 象 的 任何 线程 都 可 以 修改 这 些 对 象 。 
口 线程 调度 程序 差不多 任何 时 候 都 能 在 核心 上 调 人 或 调 出 线程 ， 
口 必须 能 调 出 运行 时 的 方法 ， 和 否则 无 限 循环 的 方法 会 一 二 占用 CPU 。 
然而 这 种 不 可 预料 的 线程 调度 可 能 会 导致 方法 “半途 而 废 "， 并 出 现状 态 不 一 致 的 对 象 。 
革 一 线程 对 数据 做 出 修改 时 ， 会 让 其 他 线程 无 法 见 到 本 应 可 见 的 修改 。 为 了 缓解 这 些 风 
险 ，Java 提 出 了 最 后 一 点 要 求 。 
口 为 了 保护 脆弱 的 数据 ， 对 人 象 可 以 被 锁 住 。 
Java 基 于 线程 和 锁 的 并 发 非常 底层 ， 并且 一 般 都 比较 难 用 。 为 了 解决 这 个 问题 ，Java 5 引信 
了 一 组 并 发 类 库 java .util .concurrent。 这 个 包 中 提供 了 一 大 ,编写 并 发 代码 的 工具 ， 很 多 程 
序 员 都 觉得 它 要 比 传统 的 块 结构 并 发 原 语 易 用 。 
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但 15 年 之 后 的 今天 ， 我 们 对 于 如 何 编写 并 发 代码 已 经 是 了 车 指 党 了 。 
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事实 证 明 ，Java 最 初 的 一 些 设 计 决 策 给 大 多 数 程序 员 编写 多 线程 代码 带 来 了 很 多 困难 。 这 
的 确 很 糟糕 , 因为 硬件 一 直 朝 着 多 核 处 理 器 方向 发 展 , 而 唯一 能 利用 好 这 些 核心 的 就 是 并 发 代 
码 。 本 章 会 讨论 一 些 在 编写 并 发 代码 时 所 通 到 的 困难 ,现代 处 理 器 对 并 发 编程 有 着 合理 的 需求 ， 
我 们 在 第 6 音 讨 论 性 能 时 还 会 涉及 其 中 的 一 些 细节 。 


随 着 开发 人 员 编写 并 发 代码 的 经 验 越 来 越 丰富 , 他 们 发 现 目 己 所 关注 的 一 些 重 要 系统 问题 一 
青 出 现 。 我 们 把 这 些 关 注 点 称 为 “设计 原则 ”一 一 存在 于 并 发 OO 系统 实际 设计 中 的 指导 性 原则 
(并 经 常 相互 冲突 )。 

在 后 面 几 节 ， 我 们 会 花 点 时 间 丁 解 一 下 其 中 最 重要 的 几 个 原则 。 


4.1.2 ”设计 理念 


Doug Lea 在 创造 他 那里 程 碑 式 的 作品 java .util.concurrent 时 列 出 了 下 面 这 些 最 重要 的 
设计 原则 : 

口 安全 性 (也 叫做 并 发 类 型 安全 性 ) 

口 活跃 度 

口 性 能 

口 重用 性 

下 面 我 们 来 逐一 解读 。 

1. 安全 性 与 并 发 类 型 安全 性 

安全 性 是 指 不 管 同 时 发 生 多 少 操作 都 能 确保 对 象 保持 目 相 一 致 。 如 果 一 个 对 象 系统 具备 这 一 
特性 ， 那 它 就 是 并 发 类 型 安全 的 。 

可 能 你 从 它 的 名 字 就 猜 出 来 了 ,并 发 可 以 看 做 是 常规 对 象 建 模 和 类 型 安全 概念 的 一 种 延伸 。 
在 非 并 发 代码 中 , 要 确保 不 管 调用 了 对 象 中 的 什么 公开 方法 对象 最 后 总 是 处 于 一 个 定义 良好 并 
上 且 一 致 的 状态 下 。 通 稼 用 来 达成 这 一 点 的 做 法 是 保证 对 象 所 有 状态 都 私有 , 并 且 开 放出 来 的 公开 
API 方 法 只 能 以 目 相 一 致 的 方式 修改 对 象 状态 

并 发 类 型 安全 的 概念 跟 对 象 类 型 安全 一 样 ， 但 它 用 在 更 复杂 的 环境 下 。 在 这 样 的 环境 中 ,其 
他 线程 在 不 同 CPU 内 核 上 同时 操作 同一 对 象 。 


人 rr ry 
何 非 私有 方法 , 而 且 也 绝 不 能 调用 其 他 任何 对 象 中 的 方法 。 如果 把 这 个 策略 跟 某 种 对 非 一 致 对 
象 的 保护 办 法 (比如 同步 镇 或 临界 区 ) 结合 起 来 ,就 可 以 保证 系统 是 安全 的 。 


2. 活跃 度 
在 一 个 活跃 的 系统 中 ， 所 有 做 出 笃 试 的 活动 最 终 或 者 取得 进展 ， 或 者 失败 。 
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这 个 定义 中 的 关键 词 是 “最 终 ” 一 运行 中 的 瞬时 故障 ( 尽管 不 理想 , 但 单独 来 看 这 不 是 问 
题 ) 和 永久 故障 是 不 同 的 。 下 面 这 几 种 底层 问题 可 能 会 导致 系统 出 现 瞬时 故障 ， 

口 处 于 锁定 状态 或 者 在 等 待 得 到 线程 锁 

口 等 待 输入 ( 比如 网 络 1/O ) 

口 资源 的 暂时 故障 

口 CPU 没 有 足够 的 空闲 时 间 运 行 该 线程 

导致 系统 出 现 永 久 故 障 的 原因 较 和 多 ， 其 中 最 常见 的 是 : 

口 死 锁 

口 不 可 恢复 的 资源 问题 ( 比如 NEFS 不 可 访问 ) 

口 信号 丢失 

尽管 你 对 它们 可 能 都 已 经 很 熟悉 了 ， 但 本 章 后 续 还 是 会 讨论 一 下 锁定 和 其 他 几 个 问题 。 

3. 性 能 

系统 性 能 可 以 通过 几 种 不 同 的 方式 量化 。 我 们 会 在 第 6 章 讨 论 性 能 分 析 和 优化 技术 , 并 且 会 介 
绍 一 些 你 应 该 了 解 的 指标 。 现 在 , 你 可 以 把 性 能 看 成 是 测量 系统 用 给 定 资源 能 做 多 少 工作 的 办 法 。 

4. 可 重用 性 

可 重用 性 是 第 四 个 设计 原则 ， 其 他 它 原则 中 并 没 涉 及 这 一 点 。 尽 管 有 时 不 容易 实现 ， 但 我 们 
还 是 非 芝 希望 能 设计 出 易于 重用 的 并 发 系统 .用 可 重用 工具 集 ( 比如 java.util.concurrent )， 
并 把 不 可 重用 的 应 用 代码 构建 在 工具 集 之 上 是 一 种 可 行 的 办 法 。 


4.1.3 ”这 些 原则 如 何以 及 为 何 会 相互 冲突 


设计 原则 经 常 相互 对 立 ， 这 种 紧张 关系 使 得 并 发 系统 的 设计 很 难 达 到 优秀 的 水 准 。 

口 安全 性 与 活 茎 度 相 互 对 立 一 一 安全 性 是 为 了 确保 坏事 不 会 发 生 ， 而 活 姥 度 要 求 见 到 进展 。 

口 可 重用 的 系统 倾向 于 对 外 开放 其 内 核 ， 可 这 会 引发 安全 问题 。 

口 一 个 安全 但 编写 方式 幼稚 的 系统 性 能 通常 都 不 会 太 好 ， 因 为 里 面 一 般 会 用 大 量 的 锁 来 保 

证 安全 性 。 
最 终 应 该 尽量 让 代码 达到 一 种 平衡 的 状态 , 使 其 能 够 灵活 地 适用 于 各 种 问题 , 却 又 能 保证 安 
全 性 ， 同 时 茜 牙 度 和 性 能 也 可 以 达到 一 定 水 平 。 这 种 境界 相当 高 ， 但 你 很 幸运 ， 我 们 马上 教 你 一 
此 实战 技巧 。 下 面 是 几 个 最 常见 的 粗浅 办 法 。 
OQ 尽 可 能 限制 于 系统 之 间 的 通信 。 了 隐藏 数据 对 安全 性 非常 有 帮助 。 
口 尽 可 能 保证 子 系统 内 部 结构 的 确定 性 。 比 如 说 ， 即 便 子 系统 会 以 并 发 的 、 非 确定 性 的 方 
式 进行 交互 ， 子 系统 内 部 的 设计 也 应 该 参照 线程 和 对 象 的 静态 知识 ， 

口 采用 客户 回应 用 必须 遵守 的 策略 方针 。 这 个 技巧 虽然 强大 ， 却 依赖 于 用 户 应 用 程序 的 合 
作 程 度 ， 并 且 如 果 某 个 精 糕 的 应 用 不 遵守 规则 ， 便 很 难 发 现 问题 所 在 。 

D 在 文档 中 记录 所 要 求 的 行为 。 这 是 最 逊 的 办 法 ， 但 如 果 代 码 要 部 署 在 非常 通用 的 环境 中 ， 
就 必须 来 用 这 个 办 法 。 
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开发 人 员 应 该 了 解 所 有 可 能 的 安全 机 制 ,而且 尽 可 能 采用 最 强 的 技术 ,但 同时 你 也 应 该 知道 ， 
在 某 些 情况 下 只 能 采用 那些 比较 还 的 办 法 。 


4.1.4 系统 开销 之 源 


并 发 系统 中 的 系统 开销 是 与 生 俱 来 的 ， 这 些 开 销 来 自 : 

口 锁 与 监测 

口 环境 切换 的 次 数 

口 线程 的 个 数 

口 调度 

口 内 存 的 局 部 性 ” 

口 算法 设计 

你 应 该 以 此 为 基础 在 大 脑 中 列 一 个 检查 列表 。 在 编写 并 发 代码 时 , 应 该 确保 自己 对 列表 中 的 
每 一 项 都 认真 考虑 过 了 ， 人 然后 再 来 “搞定 ”代码 。 


人 大 汪 pp pg 和 语言 rp 
好 的 程序 员 。 在 这 里 我 们 向 你 推荐 由 Thomas H. Corman 等 人 编著 的 《算法 导论 》( MIT，2009 ) 
和 Steven Skiena 写 的 《算法 设计 手册 》( Springer-Verlag，2008 ), 无 论 你 是 想 了 解 单线 程 算法 还 
是 想 学 习 并 发 算法 ， 它 们 都 是 值得 阅读 的 好 书 。 


本 章 会 提 到 许多 系统 开 钠 的 源头 【还 有 第 6 章 讨 论 性 能 的 部 分 )。 


4.1.5 “一 个 事务 处 理 的 例子 


本 节 前 面 的 内 容 都 太 理 论 化 了 , 所 以 我 们 补充 一 个 并 发 程序 设计 的 例子 来 实证 一 下 。 在 这 个 
例子 中 你 将 看 到 如 何 用 java.util .concurrent 中 的 高 层 类 完成 这 个 任务 。 

假设 有 一 个 基本 事务 处 理 系 统 , 构建 这 种 程序 有 个 简单 的 标准 办 法 , 就 是 先 将 业务 流程 的 不 
同 环节 对 应 到 应 用 程序 的 不 同 阶段 , 然后 用 不 同 的 线程 池 表 示 不 同 的 应 用 阶段 , 每 个 线程 池 逐 一 
接受 工作 项 ， 在 对 每 个 工作 项 进行 一 系列 的 处 理 后 ， 交 给 下 一 个 线程 池 。 通 常 来 说 ,好 的 设计 会 
让 每 个 线程 池 所 做 的 处 理 集 中 在 一 个 特定 功能 区 内 。 如 图 4-1 所 示 。 

如 果 你 设计 成 这 样 的 程序 ， 就 可 以 提高 吞吐 量 , 因为 可 以 设计 成 同时 处 理 几 个 工作 项 。 比 如 
在 检查 一 个 工作 项 的 信用 情况 时 ,可 以 检查 另 一 个 工作 项 的 库存 。 根 据 应 用 程序 的 处 理 细 节 不 同 ， 
甚至 可 以 同时 检查 多 个 订单 的 库存 。 


局 部 性 指 的 是 程序 行为 的 一 种 规律 : 在 程序 运行 中 的 短 时 间 内 ， 程序 访问 数据 位 置 的 集合 限于 局 部 范围 。 局 部 性 
有 两 种 基本 形式 : 时 间 局 部 性 与 空间 局 部 性 。 时 间 局 部 性 指 的 是 反复 访问 同一 个 位 置 的 数据 ; 空间 局 部 性 指 的 是 
反复 访问 相 邻 的 数据 。 一 一 详 者 注 
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Web 服 务 吕 


订单 接收 线程 


” 中 间 件 “ 


- 
图 4-1 才 线 程 应 用 程序 示例 


这 种 设计 非常 适合 用 java.util.concurrent 包 中 的 类 来 实现 。 这 个 包 里 有 用 于 执行 任务 
的 线程 池 ( Executors 类 中 有 一 套 工 厂 方法 可 以 创建 它们 ) 和 在 不 同 线程 池 之 间 传 递 工 作 的 队列 ， 
还 有 并 发 数据 结构 ( 可 以 用 来 构建 共享 缓存 ， 或 用 于 其 他 用 途 ) 和 很 多 其 他 底层 工具 。 

你 可 能 会 问 ， 在 Java 5 之 前 ， 还 没有 这 些 类 时 是 怎么 办 的 ? 一 般 情 况 下 ， 开 发 小 组 会 自己 编 
与 并 发 编程 类 库 ， 最 终 会 构建 出 跟 java .util.concurrent 类 似 的 组 件 。 但 这 种 定制 组 件 大 多 
存在 设计 缺陷 ， 还 会 有 难以 提 措 的 并 发 bug。 如 盯 没 有 java.util.concurrent， 开 发 人 员 就 得 
重复 实现 其 中 的 大 部 分 组 件 ( 可 能 会 有 很 多 bug， 测 试 也 不 充分 )。 

请 记 住 这 个 例子 ,我们 要 转 入 下 一 主题 一 一 温习 一 下 Java 的 “传统 ”并 发 ， 并 深入 了 解 用 它 
编程 困难 的 原因 。 


4.2 ” 块 结构 并 发 (Java 5 之 前 ) 


本 董 大 部 分 内 容 都 在 讨论 块 同步 并 发 方式 的 替代 方案 。 如 果 你 想 从 我 们 的 讨论 中 获 益 ,就 需 
要 深刻 理解 传统 并 发 的 优 缺 点 的 重要 性 。 

为 此 我 们 要 讨论 用 到 svnchronizea、volatile 等 并 发 关键 字 的 那 种 原始 、 低 级 的 多 线程 
编程 方式 。 我 们 把 这 个 讨论 放 在 设计 原则 的 情境 中 ， 并 且 会 着 眼 于 下 一 节 将 要 讨论 的 内 容 。 

之 后 我 们 会 简略 地 解释 一 下 线程 的 生命 周期 , 然后 讨论 常见 的 并 发 编程 技巧 和 陷阱 ， 比 如 完 
全 同步 的 对 象 ， 死 锁 ，volatile 关 键 字 和 不 变性 。 

我 们 先 重 温 一 下 同步 吧 。 
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4.2.1 同步 与 锁 


你 知道 的 ，synchronized 既 可 以 用 在 代码 块 上 也 可 以 用 在 方法 上 。 它 表明 在 执行 整个 代码 
块 或 方法 之 前 线程 必须 取得 合适 的 锁 。 对 于 方法 而 言 ， 这 意味 着 要 取得 对 象 实 例 锁 ( 对 于 静态 方 
法 而 言 则 是 类 锁 )。 对 于 代码 块 ， 程 序 员 则 应 该 指明 要 取得 哪个 对 象 的 锁 。 

在 任何 一 个 对 象 的 同步 块 或 方法 中 ,每 次 只 能 有 一 个 线程 进入 ;如果 其 他 线程 试图 进入 ,JVM 
会 挂 起 它们 。 无 论 其 他 线程 试图 进入 的 是 该 对 彰 的 同一 同步 块 还 是 不 同 的 同步 块 ，JVM 各 会 如 此 
处 理 。 这 种 结构 在 并 发 理论 中 被 称 为 临界 区 ， 


注意 ”你 有 没有 想 过 Java 中 用 于 确立 临界 区 的 关键 字 为 什么 是 synchronized? 为 什么 不 是 
“critical” 或 “locked”?” 同步 的 是 什么 ? 我 们 会 在 4.2.5 节 回 到 这 一 话题 上 来 ， 但 如 果 你 
不 知道 ， 或 者 从 来 没 想 过 这 个 问题 ， 你 最 好 花 几 分 钟 想 一 想 再 继续 。 


本 草 要 讨论 一 些 比较 新 的 并 发 技术 。 但 网 然 说 到 了 同步 ， 我 们 就 顺便 看 看 与 Java 中 的 同步 和 
锁 相关 的 一 些 基本 事实 吧 。 希望 你 对 这 里 的 大 多 数 (或 全 部 ) 知识 都 已 经 烂熟 于 心 了 。 

口 只 能 锁定 对 象 ， 不 能 锁定 原始 类 型 。 

口 被 锁定 的 对 象 数 组 中 的 单个 对 象 不 会 被 锁定 。 

口 同步 方法 可 以 视 同 为 包含 整个 方法 的 同步 (this) { .,，1 代 码 块 (但 要 注意 它们 的 二 
进 制 码 表示 是 不 同 的 ) 

口 静态 同步 方法 会 锁定 它 的 class 对象， 因为 设 有 实例 对 象 可 以 锁定 。 

口 如 果 要 锁定 一 个 类 对 象 ， 请 慎重 考虑 是 用 显 式 锁定 ,还 是 用 getclass () ， 两 种 方式 对 子 
类 的 影响 不 同 。 

口 内 部 类 的 同步 是 独立 于 外 部 类 的 ( 要 明白 为 什么 会 这 样 ， 请 记 住 内 部 类 是 如 何 实现 的 )。 

口 synchronized 并 不 是 方法 签名 的 组 成 部 分 ， 所 以 不 能 出 现在 接口 的 方法 声明 中 ，。 

口 非 同 步 的 方法 不 查看 或 关心 任何 锁 的 状态 ， 而 且 在 同步 方法 运行 时 它们 仍 能 继续 运行 。 

口 Java 的 线程 锁 是 可 重信 的 。 也 就 是 说 持 有 锁 的 线程 在 遇 到 同一 个 锁 的 同步 点 ( 比如 一 个 同 
步 方法 调用 同一 个 类 内 的 男 一 个 同步 方法 ) 时 是 可 以 继续 的 。 


警告 ”在 其 他 语言 中 存在 不 可 重 入 的 锁 机 制 ( 用 java 也 能 实现 相同 的 效果 , 如果 你 想 了 解 那些 让 
人 看 了 万 骨 恒 然 的 详细 信息 , 请 参见 java.util.concurrent .locks 中 的 ReentrantLock 
的 Javadoc )， 但 和 它们 打交道 太 痛 苦 了 ， 除 非 你 真 的 知道 自己 在 做 什么 ， 和 否则 还 是 躲 之 
为 好 。 


对 Java 同 步 的 温习 就 到 此 为 止 吧 。 现 在 我 们 来 看 一 下 线程 在 其 生命 周期 中 的 状态 变迁 。 
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4.2.2 线程 的 状态 模型 


图 4-2 展 示 了 线程 生命 周期 的 发 展 过 程 一 一 从 创建 到 运行 ， 到 青 次 运行 之 前 (或 被 资源 阻 窄 ) 
可 能 被 挂 起 ， 再 到 最 终 完成 。 
Java 


Obiject notify(}s 
Object notifyAll(}):; 


本 Et (js 


: 收 到 数据 /同步 


广 LO 或 | 同 | | 
步 阻塞 | 其 他 线程 关闭 了 套 接 字 


图 4-2 _ Java 的 线程 状态 模型 


线程 最 初创 建 时 处 于 就 绪 ( Ready ) 状态 。 然 后 调度 器 会 找 个 核心 来 运行 它 ， 如 果 机 器 负载 
过 重 , 那 它 就 可 能 需要 多 些 时 间 。 开 始 运行 之 后 ， 线 程 通 常会 消耗 掉 分 配给 它 的 时 间 ， 然 后 回 到 
就 绪 状 态 , 等 到 下 次 再 有 处 理 硕 分配 时 间 片 给 它 。 这 是 我 们 在 4.1.1 节 提 过 的 抢占 式 线程 调度 的 标 
准 动作 。 

除了 由 调度 器 发 起 的 标准 动作 , 线程 本 身 也 能 表明 它 此 时 无 法 使 用 核心 工作 。 这 可 能 是 因为 
程序 代码 通过 Thread. sleep () 告 诉 线程 在 继续 之 前 应 该 暂停 , 或 者 因为 线程 必须 等 待 通知 ( 通 
常 需要 满足 某 些 外 部 条 件 )。 这 时 线程 会 从 核心 中 称 走 ， 并 释放 它 持 有 的 锁 。 只 有 通过 吃 醒 才能 
青 次 运行 线程 (在 达到 睡眠 时 长 之 后 ， 或 收 到 了 恰当 的 信和 号 )， 进 人 就 绪 状 态 。 

线程 可 能 会 因为 等 待 UO 或 等 待 效 取 其 他 线程 持 有 的 锁 而 被 阻塞 。 这 时 线程 并 没有 被 交换 出 
核心 ， 而 是 仍然 处 于 繁忙 状态 ， 等 着 获取 可 用 的 锁 或 数据 。 在 得 到 锁 或 数据 之 后 ,线程 会 继续 执 
行 直到 它 的 时 间 片 结束 。 

我 们 接 下 来 讨论 一 个 著名 的 解决 同步 问题 的 办 法 一 一 完全 同步 对 象 。 


4.2.3 ”完全 同步 对 象 


前 面 介绍 了 并 发 类 型 安全 的 概念 , 还 提 到 了 一 种 用 来 达成 这 种 安全 性 的 策略 ( 在 “保证 安全 ” 
的 边栏 中 )。 现 在 我 们 来 看 一 下 这 个 策略 更 完整 的 描述 ， 它 通常 被 称 为 完全 同步 对 象 。 如 果 一 个 


3 者 
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类 遵从 下 面 所 有 规则 ， 就 可 以 认为 它 是 线程 安全 并 且 活 路 的 。 

一 个 满足 下 面 所 有 条 件 的 类 就 是 完全 同步 类 。 

口 所 有 域 在 任何 构造 方法 中 的 初始 化 都 能 达到 一 致 的 状态 。 

口 没有 公共 域 。 

口 从 任何 非 私有 方法 返回 后 ， 都 可 以 保证 对 象 实例 处 于 一 致 的 状态 (假定 调用 方法 时 状态 

是 一 致 的 )。 

口 所 有 方法 经 证 明 都 可 在 有 限时 间 内 终止 。 

口 所 有 方法 都 是 同步 的 。 

口 当 处 于 非 一 致 状态 时 ， 不 会 调用 其 他 实例 的 方法 。 

口 当 处 于 非 一 致 状态 时 ， 不 会 调用 非 私 有 方法 。 

假定 有 一 个 分 布 式微 博 工具 ， 代 码 清单 4-1 是 其 后 台中 的 类 。 在 它 的 propagateUpdate 1) 
方法 被 调用 时 , ExampleTimingNode 类 会 收 到 更 新 , 也 可 以 通过 查询 看 它 是 否 收 到 了 特定 更 新 。 
这 是 经 典 的 读 写 操作 相互 冲 罕 的 情景 ， 需 要 通过 同步 防止 出 现 不 一 致 状态 。 


public class ExampleTimingNode implements SimpleMicroBlogNode | 


private final String identifier:; , 二 
Pa A 
private final Map<Update, Longs arrivalTime 没有 公开 域 


se = new HashMap<>().:; 


publie ExampleTimingNode (String identifier ) 1 J] 所有 酝 丰 机 1 
identifier = identifier ; der 


public synchronized String getIidentifier() 1 
return ldentifier; 


} 
public synchronized void propagateUpdate I 
“= Update update ) I 
long currentTime = SYystem.currentTimeMill1isl); 
arrivalTime.put (update , currentTime),; 
} 
public synchronized boolean confirmUpdateReceived | 
ww Update update ) | 
Long timeRecvd = arrivalTime .get lupdate ): 
return timeRecvd l= null; 
} 
} 
这 是 一 个 既 安 全 又 活 唉 的 类 ， 第 一 眼看 上 去 让 人 感觉 很 了 不 起 。 但 随 之 而 来 的 是 性 能 问 
题 , 既 安 全 又 酒 跌 的 东西 速度 不 一 定 也 能 很 快 ,必须 用 synchronized 去 协调 对 Map arrival- 
Time 的 所 有 访问 ( get 和 put )， 而 这 个 锁 最 终 会 把 你 的 速度 拖 慢 。 这 是 并 发 处 理 方 式 的 主要 
问题 。 


所 有 方法 都 是 同 
步 的 


1 二 
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除了 性 能 问题 ， pmpvgWepwrovr 脆 梯 。， 从 来 不 会 在 同 
aLrivalnime， 实 际 上 只 是 调用 get 和 Dut 者 滨 、 但 这 只 有 在 代码 量 很 小 名 : 
在 真实 的 大 型 系统 中 ， 代码 太 多 而 无 法 实现 这 种 方法 。 同 时 ， bug 也 很 容易 潜伏 在 庞大 的 4 
库 中 ， 这 也 是 开始 寻求 更 完善 的 解决 方法 的 另 一 个 原因 。 让 


4.2.4 死 锁 
并 发 的 另 一 个 经 典 问题 是 死 镇 。 代 码 清单 4-2 稍 微 扩 展 了 一 下 上 个 例子 。 在 这 一 版 中 ， 除 了 
记录 最 近 一 次 更 新 的 时 间 ， 每 个 节 扣 收 到 更 新 时 还 会 通 和 为 外 一 个 廊 点 。 
这 有 段 代码 试图 构建 一 个 多 线程 的 更 新 处 理 系统 。 注 意 , 这 有 段 代码 是 为 了 解释 死 锁 ,不 要 把 它 
用 到 你 的 工作 中 。 
代码 清单 4-2 死 锁 的 例 - 


Public class MicroBlogNode implements SimpleMicroBlogNode | 
private final String ident.:; 


public MicroBlogNode (String ident ) | 
ident = ident } 


| 


public String getIdent() 1 
return ident,; 
} 
public synchronized void propagateUpdate (Update upd , MicroBlogNode 
backup ) | 
System.out .printlnlident +": recvd: "+ Upd .getUpdateText () 
a +" ;} backup: "+backup .getIdent())},; 
backup .confirmUpdate (this, upd ); 
} 
public synchronized void confirmUpdate (MicroBlogNode other , Update 
update ) 1 
System.out .println(ident +": recvd confirm: "+* 
= Update .getUpdateText() +" from "+0other .getIdent (}k}):; 
) 
| 
final MicroBlogNode local = | 关键 字 finali 是 必需 的 
= new MicroBlogNodel"localhost :B888").; | 
final MicroBlogNode other = new MicroBlogNode ("localhost:8988"):; 
final Update firaet = getUpdate ("1"), 
final Update second = getUpdate("2"}).; 


new Thread{(new Runnable() | | 第 一 个 更 新 发 送 
public void run() 1 给 第 一 个 线程 
local .propagateUpdate (first, other):; 中- 一 一 


| 
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}) .start ()，; 


new Thread(new Runnable{() 1 第 二 个 更 新 发 送 
public void run() 1 给 第 二 个 线程 
other.propagateUpdate (second, local); 


1}) .start (); 

乍 一 看 ,这 段 代码 没什么 毛病 。 有 两 个 更 新 分 别 发 送 给 不 同 的 线程 ， 每 个 都 必须 由 后 备 线程 

进行 确认 。 这 看 起 来 不 是 什么 离奇 古怪 的 设计 一 一 如 果 一 个 线程 失效 , 另外 一 个 线程 还 可 以 挑 起 

如 果 你 运行 这 段 代 码 , 一 般 都 会 磁 到 死 锁 一 一 两 个 线程 都 说 自己 收 到 了 更 新 , 但 它 俩 谁 都 不 

会 以 备份 线程 的 身份 确认 收 到 了 更 新 。 因 为 每 个 线程 在 确认 方法 能 够 确认 之 前 都 要 求 另外 一 个 线 
程 释放 线程 锁 ， 如 图 4-3 所 示 。 

线程 | 线程 2 


Acauire (A) 


图 4-3 ” 死 锁 线程 


有 一 个 处 理 死 锁 的 技巧 ， 就 是 在 所 有 线程 中 都 以 相同 的 顺序 获取 线程 锁 。 在 前 例 中 , 第 一 个 
线程 以 A、B 的 顺序 获取 锁 , 而 第 二 个 线程 获取 锁 的 顺序 是 B、A。 如 果 两 个 线程 都 用 A . B 的 顺序 ， 
死 锁 的 情况 就 可 以 避免 ， 因 为 第 二 个 线程 在 第 一 个 线程 完成 并 释放 锁 之 前 会 直 被 阻塞 住 。 

就 完全 同步 对 象 方式 而 言 ， 要 防止 这 种 死 锁 出 现 是 因为 代码 破坏 了 状态 一 致 性 规则 。 当 有 消 
息 到 达 时 ， 接 受 节 点 会 在 消息 处 理 过 程 中 调用 男 外 一 个 对 象 一 一 它 发 起 这 个 调用 时 状态 是 不 一 
致 的 。 

接 下 来 ， 我们 会 返回 来 解释 前 面 抛 出 的 那个 问题 ; 为 什么 Java 中 用 来 标识 临界 区 的 关键 字 是 
synchronized? 这 会 引 守 我 们 转 而 讨论 不 可 变性 和 关键 字 volatile。 


4.2.5 为 什么 是 synchronized 

最 近 几 年 并 发 编程 变化 最 大 的 是 硬件 领域 。 在 以 前 , 程序 员 可 能 常年 累 月 都 碰 不 到 需要 支持 
多 处 理 衙 核心 ( 两 个 或 最 多 三 个 ) 的 系统 。 因 此 并 发 编程 过 去 主要 考虑 如 何 分 享 CPU 时 间 一 一 线 
程 们 在 单 核 上 轮流 上 位 ， 相 互 调换 。 
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岗 如 今 ， 任 何 比 手机 大 点 儿 的 东西 都 是 多 核 的 ， 所 以 我 们 的 认 知 模型 也 该 换 换 了 ,应 该 把 多 
个 线程 在 同一 物理 时 刻 运行 在 不 同 核 心 ( 并 且 很 可 能 会 操作 共享 的 数据 ) 的 情况 也 考虑 在 内 。 如 
图 4-4 所 示 。 为 了 提高 效率 ， 同 时 和 运行 的 每 个 线程 可 能 者 会 有 它 正 在 处 理 的 数据 的 缓存 复 本 。 记 
住 这 幅 图 .让 我 们 回 到 选择 用 什么 关键 字 来 表示 被 锁定 的 代码 块 或 方法 这 个 问题 上 。 


程序 人 程序 B 


程序 A 程序 B 


CPU 


图 4-4 考虑 并 发 和 线程 的 新 、 老 方式 
我 们 在 前 面 问 过 ， 代 码 清 单 4-1 中 被 同步 的 是 什么 ”等 案 是 : 被 同步 的 是 在 不 同 线 程 中 表示 
被 锁定 对 章 的 内 看 块 。 也 就 是 说 ， 在 synchronized 代 码 块 (或 方法 ) 执行 完 之 后 ， 对 被 锁定 对 
象 所 做 的 任何 修改 全 部 蹇 会 在 线程 锁 释 放 之 前 刷 回 到 主 内 存 中 ， 如 图 4-5 所 示 : 


人 | “被 同步 的 内 存 块 


图 4-5 不 同 线程 对 一 个 对 象 的 修改 通过 主 内 存 传 播 
另外 ,， 当 进入 一 个 同步 的 代码 块 ， 得 到 线程 锁 之 后 ， 对 被 锁定 对 象 的 任何 修改 都 是 从 主 内 存 
中 读 出 来 的 , 所 以 在 锁定 区 域 代码 开始 执行 之 前 , 持 有 锁 的 线程 就 和 锁定 对 象 主 内 存 中 的 视图 同 
步 了 。 


4.2.6 ”关键 字 volatile 


Java 在 其 混沌 初 开 的 时 期 ( Java 1.0 ) 就 已 经 把 volactile 作 为 关键 字 了 ， 它 是 一 种 简单 的 对 
象 域 同步 处 理 办 法 ,包括 原始 类 型 。 一 个 volatile 域 需 遵循 如 下 规则 

口 线程 所 见 的 值 在 使 用 之 前 总 会 从 主 内 存 中 再 读 出 来 。 

口 线程 所 写 的 值 总 会 在 指令 完成 之 前 被 刷 回 到 主 内 存 中 。 
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可 以 把 围绕 该 域 的 操作 看 成 是 一 个 小 小 的 同步 块 。 程序 员 可 以 便 此 编写 简化 的 代码 , 但 付出 
的 代价 是 每 次 访问 都 要 额外 刷 一 次 内 存 。 还 有 一 点 要 注意 ，volatile 变 量 不 会 引信 线 程 锁 ， 所 
以 使 用 volatile 变 量 不 可 能 发 生死 锁 。 

更 加 微妙 的 是 ，volatile 变 量 是 真正 线程 安全 的 ， 但 只 有 写 人 时 不 依赖 当前 状态 (该 取 的 
状态 ) 的 变量 才 应 该 声明 为 volatile 变 量 。 对 于 要 关注 当前 状态 的 变量 ， 只 能 借助 线程 锁 保 证 
其 绝对 安全 性 。 


4.2.7 不 可 变性 


不 可 变 对 象 的 应 用 是 十 分 有 价值 的 技术 。 这 些 对 象 或 没有 状态 , 或 内 有 final 域 (因此 只 能 
在 构造 方法 中 赋值 )。 它们 总 是 安全 而 又 活路 的。 它们 的 状态 不 能 修改 ， 所 以 不 可 能 出 现 不 一 致 
的 情况 。 

可 这 样 对 象 初始 化 的 所 有 值 都 必须 传人 构造 方法 。 这 会 导致 构造 方法 的 参数 很 多 , 看 起 来 又 
春 又 繁 。 因 此 很 多 程序 员 选 择 工 厂 方法 FactoryMethod 代 替 构 造 方法 。 工 厂 方法 很 简单 ， 就 是 
类 中 的 一 个 静态 方法 ， 用 来 代 蔡 构造 方法 创建 新 对 象 。 此 时 构造 方法 通常 被 声明 为 protected 
或 private 的 ， 从 而 使 工厂 方法 成 为 实例 化 对 象 的 唯一 办 法 。 

但 是 还 存在 要 将 众多 参数 传人 FactorvyMethod 的 问题 。 有 时 候 这 不 太 方 便 ， 尤 其 是 初始 化 
对 象 所 需 的 状态 参数 有 多 个 不 同 来 源 时 。 

构建 器 模式 可 以 解决 这 个 问题 。 它 由 两 部 分 组 成 : 一 个 是 实现 了 构建 器 泛 型 接口 的 内 部 静态 
类 ， 另 一 个 是 构建 不 可 变 类 实例 的 私有 构造 方法 。 

内 部 静态 类 是 不 可 变 类 的 构建 着 , 开发 人 员 只 能 通过 它 获 取 不 可 变 类 的 新 实例 。 比 较 常 见 的 
实现 方式 是 让 构建 器 类 拥有 与 不 可 变 类 一 模 一 样 的 域 ， 但 构建 器 的 域 是 可 修改 的 。 

下 面 这 段 代码 展示 了 如 何 建 立 不 可 变 的 微 博 更 新 模型 ( 根据 本 章 前 面 的 例子 所 构建 )。 


代码 清单 4-3 ”不 可 变 对 象 及 构建 器 


public interface ObijBuilder<T> | 到 
T buildl); 构建 器 接口 
| 


public class Update | 


private final Author author; | 必须 在 构造 方法 中 官 始 化 
private final String updateText; | final 


private Update (Builder b ) i1 
author = b .author; 
updateText = b .updateText; 


构造 器 类 必须 是 静态 
Fublic static class Builder 内 部 类 
a imnlements ObjBuilder<Updates | 


private Author author:; 
private String updateText; 


1 二 
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public Builder author (Author author ) { pe 
author = author ; 可 用 在 调用 链 中 返回 
return this; Builder 的 方法 


} 


public Builder updateText (String updateText ) | 
UpdateText = UpdateText ; 
return this; 


public Update build(}) { 


return new Update (this),) 畴 去 hashcode() 和 
| equalsa() 方 法 


| 
有 了 这 段 代码 ， 你 就 可 以 创建 新 的 Upaate 对 象 ， 


Update.Builder ub = new Update .Bui1ladeTr1l : 
Update U = Ub.author (myAuthor) .updateText ("Hello") ,build(}; 


这 是 一 个 得 到 广泛 应 用 的 通用 模式 。 实 际 上 ， 在 代码 清单 4-1 和 4-2 中 我 们 已 经 使 用 了 不 可 变 
对 象 的 特性 。 

关于 不 可 变 对 象 的 最 后 一 点 一 一 关键 字 final 仅 对 其 直接 指向 的 对 象 有 用 。 如 图 4-6 所 示 
对 主 对 象 的 引用 不 能 研 值 为 对 象 3， 但 在 主 对 象 内 部 ， 对 1 的 引用 可 以 改 为 指向 对 象 2。 也 就 是 
final 弓 [用 可 以 指 同市 有 非 tinal 域 的 对 象 。 


时 


图 4-6 值 的 不 可 变性 与 引用 


不 可 变 是 非常 强 的 技术 ,用 处 十 分 广泛 。 但 有 时 候 只 用 不 可 变 对 象 开 发 效率 不 行 ， 因 为 每 次 
修改 对 象 状态 就 需要 构建 一 个 新 对 象 。 所 以 可 变 对 象 很 有 必要 保留 。 

我 们 马上 就 要 开始 讨论 本 章 最 重要 的 主题 一 一 java .util .concurrent 中 更 加 现代 化 、 概 
念 更 简单 的 并 发 API。 看 看 怎么 用 它们 写 代码 。 


4.3 现代 并 发 应 用 程序 的 构件 


随 着 Java 5 的 到 来 , Java 对 并 发 的 重新 思考 也 浮 出 了 水 面 ,这 些 新 思想 主要 体现 在 java.util. 
concurrent 包 上 ， 其 中 包含 了 大 量 用 来 编写 多 线程 代码 的 新 工具 。 在 后 续 版 本 中 ， 这 些 工具 不 
新 得 到 改进 ， 但 其 工作 方式 却 依然 保持 不 变 ， 并 且 和 直到 今天 还 是 对 开发 人 员 很 有 帮助 。 

我 们 马上 快速 过 一 下 java.util.concurrent 中 主要 的 类 及 相关 包 ， 比 如 atomic 和 locks 包 。 
我 们 会 向 你 介绍 这 些 类 及 其 适用 的 情景 。 你 也 应 该 读 一 下 它们 的 Javadoc, 并 尝试 熟悉 整个 包 
它们 使 编写 并 发 类 容易 多 了 。 
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i 。 按 我 们 的 经 验 ， 几 乎 在 所 有 人 案例 中 ， 如 果 你 特意 : rs 码 迁 移 到 新 的 API 中 ， 
代码 就 会 得 以 改进 。 你 的 努力 付出 将 使 代码 在 清晰 性 和 可 靠 性 上 得 到 极 大 提升 。 


请 把 这 次 讨论 当做 并 发 编程 的 启动 工具 ， 而 不 是 一 次 研讨 会 。 想 要 充分 利用 好 java .util. 
concurrent ， 你 还 需要 知道 更 多 的 知识 。 


4.3.1 原子 类 : java .util.concurrent ,atomiec 


java.util.concurrent .atomic 中 有 几 个 名 字 以 Atomic 打 头 的 类 。 它 们 的 语义 基本 上 和 
volatile 一 样 ， 只 是 封闭 在 一 个 API 里 了 ， 这 个 API 包 含 为 操作 提供 的 适当 的 原子 ( 要 人 么 不 做 ， 
要 做 就 全 做 ) 方 法 。 对 于 开发 人 员 来 说 , 这 是 非常 简单 的 避免 在 共享 数据 上 出 现 竟 争 和 危害? 的 办 法 。 

在 编写 这 些 实现 时 利用 了 现代 处 理 需 的 特性 , 所 以 如 果 能 从 硬件 和 操作 系统 上 得 到 适当 的 支 
持 ， 它 们 可 以 是 非 阻 塞 ( 无 需 线 程 锁 ) 的 ， 而 大 多 数 现代 系统 都 能 提供 这 种 支持 。 常 见 的 用 法 是 
实现 序列 号 机 制 ， 在 atomicInteger 或 ALomicLong 上 用 原子 操作 getandIncrement () 方 法 。 

要 做 序列 号 ， 该 类 应 该 有 个 nextIa() 方 法 ， 每 次 调用 时 肯定 能 返回 一 个 唯一 并 且 完 全 增长 
的 数值 。 这 和 数据 库 里 序列 号 的 概念 很 像 ( 所 以 这 个 变量 叫 这 个 名 字 ) 

来 看 一 段 产生 序列 号 的 代 伍 : 


private final AtomicLong sequenceNumber = new AMAtomicLong'(0),; 


public long nextId() { 
return sequenceNumber .gethndlncrement (|); 


} 


注意 ”原子 类 不 是 从 有 相似 名 种 的 类 继承 而 来 的 ， 所 以 AtomicBoolean 不 能 当 Boolean 用 ， 
AtomicInteger 也 不 是 Integer， 曼 然 它 确实 扩展 了 Number。 


接 下 来 ,我 们 会 检查 一 下 java .util.concurrent 如 何 对 同步 模型 的 核心 建 模 Lock 接 口 。 


4.3.2 ”线程 锁 : java.util.concurrent .locks 


块 结构 同步 方式 基于 锁 这 样 一 个 简单 的 概念 。 这 种 方式 有 几 个 缺点 。 
UU 锁 只 有 一 种 类 型 。 
口 对 被 锁 住 对 象 的 所 有 同步 操作 都 是 一 样 的 作用 。 


中 竞争 危害 ( race hazard ) 又 名 竞 态 条 件 ( race condition )。 一 个 系统 或 进程 的 输出 ， 使 赖 于 不 受 控制 事件 的 出 现 顺 
序 或 时 机 。 例 如 两 个 进程 都 试图 修改 一 个 共享 内 存 的 内 容 。 在 设 有 并 发 控制 的 情况 下 ， 最 后 的 结果 取决 于 两 个 进 
程 的 执行 顺序 与 时 机 ， 如 果 发 生 了 和 并 发 访问 冲 罕 ， 最 后 的 结果 是 不 正确 的 。 一 一 详 者 注 
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口 在 同步 代码 块 或 方法 开始 时 取得 线程 锁 。 
口 在 同步 代码 块 或 方法 结束 时 释放 线程 锁 ， 
口 线程 或 者 得 到 锁 ， 或 者 阻 寨 一 一 没有 其 他 可 能 。 
如 果 我 们 要 重 构 对 线程 锁 的 支持 ， 有 几 处 可 以 得 到 提升 。 
口 添加 不 同类 型 的 锁 ， 比 如 该 取 锁 和 写 人 锁 。 
口 对 锁 的 阻塞 没有 限制 ， 即 允许 在 一 个 方法 中 上 锁 ， 在 男 一 个 方法 中 解锁 。 
口 如 果 线 程 得 不 到 锁 ， 比 如 锁 由 男 外 一 个 线程 持 有 ， 就 多 许 该 线程 后 退 或 继续 执行 ， 或 者 
做 点 别 的 事情 一 一 运用 tryLock() 方 法 。 
口 允许 线程 尝试 取 锁 ， 并 可 以 在 超过 等 待 时 间 后 放弃 。 
能 实现 以 上 这 些 的 关键 就 是 java .util .concurrent .locks 中 的 Lock 接 口 。 还 有 它 的 两 个 
实现 类 ， 
UD ReentrantLock 本 质 上 跟 用 在 同步 块 上 那 种 锁 是 一 样 的 ， 但 它 要 稍微 灵活 点 儿 。 
口 ReentrantReadWriteLock 一 一 在 需要 读 取 很 多 线程 而 写 人 很 少 线程 时 ， 用 它 性 能 会 
更 好 。 
块 结构 并 发 能 实现 的 所 有 功能 都 可 以 用 Lock 接 口 实现 。 下 面 是 用 ReentrantLock 重 与 的 那 
个 死 锁 的 例子 。 
死 锁 


代码 清单 4-4 用 ReencrantLock 重 写 


private final Lock lock = new ReentrantLock|(); 


public void propagateUpdate (Update upd , MicroBlogNode backup ) | 


lock .lock1l) ; : z | 

i 每 个 线程 都 先 锁 

志和 住 自己 的 锁 
System.out .println(ident +": recvd: "+ HJ 


= UPd .getUpdateText() +" ; backup: "+ 
sm“ backup .SetIQent () ) ; 


backup .confirmUpdate (this, upd ) ; s 
} finally { ] 调用 confirm0pdate() 
lock .unlock(); 知悉 其 他 线程 
| 
| 


public void confirmUpdate (MicroBlogNode other ，Update upd ) | 
lock. lock(); 
tryl 
System.out .println(iden +": recvd contirm: "+ 
= UpPd .getUpdateText(}) +" from "+ other .getIidentifier (})}; 
} finally 1 
lock .unlock|(); 
| 尝试 锁 住 其 他 线程 | 
锁 住 其 他 线程 的 尝试 @ 通 常 都 会 失败 ， 因 为 它 已 经 被 锁 住 了 ( 如 图 4-3 所 示 )。 这 就 是 导致 死 
锁 出 现 的 原因 。 
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机 画 本 “二 再 = le 
| J ] 2 内 让 trp : 嫩 
el —— sub 上 | es a 
. he | 一 可 二 a 1 "i 
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把 lock() 放 在 try...finally 块 中 (释放 也 在 这 里 ) 的 模式 是 另外 一 个 好 用 的 小 工具 。 
在 跟 块 结构 并 发 相似 的 情景 中 它 同 样 很 好 用 。 而 另 一 方面 ， 如 果 需 要 传递 Lock 对 人 象 ， 比 如 从 
一 个 方法 中 返回 ， 则 不 能 用 这 个 模式 。 . 
使 用 Lock 对 象 可 能 要 比 块 结构 方式 强大 得 多 , 但 有 时 用 它们 很 难 设计 出 完善 的 锁定 策略 。 


对 付 死 锁 的 策略 有 很 多 , 但 你 应 该 特别 注意 一 个 不 起 任何 作用 的 策略 。 请 看 下 面 这 段 代 码 中 
新 版 的 propagateUpdate |() 方 法 (假定 confirmUpdate() 也 做 出 了 同样 的 修改 )。 在 这 个 例子 
中 , 我 们 用 带 有 超时 机 制 的 LryLock 1) 替换 了 无 条 件 的 锁 , 通过 这 种 办 法 可 以 为 其 他 线程 提供 得 
到 线程 锁 的 机 会 ， 从 而 去 除 死 锁 。 


| EE : 二 a = 本 = 1 四 i Pn 本 | Eh | Fr 有 的 | = [a 下 a : | | 
代码 清单 4.5 “一 次 有 缺陷 的 解决 死 锁 问 题 的 尝试 


public void propagateUpdate (Update upd , MicroBlogNode backup ) | 
boolean acguired = false; 


while (!acquired}) | | : 
try | | 尝试 与 锁定 ， 
int wait = (int) (Math.random() *» 10); 超时 时 长 随机 


acquired = lock.tryLock (wait, TimeUnit .MILLISECONDS).; 
if (acauired) | 
System.out .println(lident +": recvd: "+ 
= Upd .getUpdateText () + ; backup: "+backup .getIdent |()):; 


backup .confirmUpdate (this, update ); - es 
} else | ] 在 其 他 线程 
Thread.sleep (wait) ; 上 确认 


上 catch (InterruptedException e) | 
} finally 1 


if (acquired) lock .unlock'|(); 2 
} 仅 在 锁定 
时 解锁 


} 

如 果 运 行 代码 清单 4-5 中 的 代码 ， 你 会 发 现 它 有 时 候 还 是 不 能 解决 死 锁 问题 。 你 能 看 到 
“received confirm of update"， 但 它 并 不 会 一 直 出 现 ， 时 有 时 无 。 

实际 上 , 死 锁 问 题 并 没有 真正 解决 , 因为 如 果 线 程 取得 了 第 一 个 锁 ( 在 propagateUpdate1) 
中 ), 它 才 会 调用 confirmUpdate(), 并 且 在 完成 之 前 绝 不 会 释放 第 一 个 锁 。 即 使 两 个 线程 都 能 
在 彼此 调用 confirmUpaate1() 之 前 取得 第 一 个 线程 锁 ， 它 们 还 是 会 产生 死 锁 。 

如 果 取 得 第 二 个 锁 的 尝试 失败 , 能 真正 解决 问题 的 办 法 是 让 线程 释放 其 持 有 的 第 一 个 锁 ， 再 
次 从 头 开始 等 待 ， 从 而 使 其 他 线程 有 机 会 得 到 完整 的 锁 集 合 ， 能 走 完全 程 。 代 码 如 下 所 示 。 


public void propagateUpdate (Update upd , MicroBlogNode backup ) | 
boolean acgquired = falge:; 
boolean done = false; 
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while (!'done) 1 
int wait = (int) (Math.random{() * 10);} 
try | 
acoquired = lock.tryLock (wait, TimeUnit .MILLISECONDS):; 
if lacquired) | 
System.out .println(lident +": recVvd: "+ 
mp Upd .getUpdateText () +" };} backup: A 


done = backupNode .tryconfirmUpdate (this, Update ); + 

| 检查 
} catch (InterruptedException e) | ryYContirn 
} finally 1 的 返回 什 

if (acgquired) lock .unlockl(); 
} 
if (idone) try | 

Thread .sleepnp (wait).:; 如 果 done 为 false， 
| catch (InterruptedException e) { |} ,| | 了 释放 锁 间 等 待 


} 
} 
Public boolean tryConfirmUpdate {MicroBlogNode other , Update upd ) | 
boolean acquired = false; 
try | 
int wait = (int) [Math.random(}) 二 10] ; 
acgquired = lock.tryLock (wait, TimeUnit .MILLISECONDS)., 
if (acquired) 1{ 
long elapsed = System.currentTimeMillis(} -= startTime; 
Syastem.out .println(ident +": recvd confirm: “+ 
“= upd .getUpdateText () +" from "+0ther .getIdent{() 
mb +" - took "+ elapsed +" millis"),; 
return true,; 


) 

catch (InterruptedException e@) 1 
finally | 

if (acquired) lock.unlock'():; 


| 


return falgsge:; 


} 

这 一 版 会 检查 tryconfirmUpdate() 的 返回 码 。 如 果 为 false, 最 初 的 锁 被 释放 。 该 线程 会 
暂停 一 段 时 间 ， 让 其 他 线程 有 机 会 获取 锁 。 

把 这 有 段 代码 运行 几 次 ， 你 会 发 现 这 两 个 线程 基本 上 总 能 走 完全 程 灶 问 题 已 经 被 你 解决 
了 。 你 也 许 想 试 验 试验 之 前 版 本 中 那 段 代 码 的 不 同形 式 , 诸如 最 原始 的 、 有 缺陷 的 或 被 改正 的 。 通 
过 对 这 些 代码 的 演练 ,你 能 对 锁 机 制 有 更 深刻 的 理解 ,并 且 开始 渐渐 地 赁 直觉 避免 死 锁 问题 的 出 现 。 


Tm 


ey py 那 是 什 么 原因 导致 这 个 版 本 中 的 码 有 | 人 威 功 呢 ? 1 
码 中 附加 的 复杂 性 是 罪魁 福 首 。 它 影响 JVM 的 线程 调度 器 ， 让 它 变 得 更 加 难以 预测 。 这 意味 着 
它 有 时 候 能 让 某 个 线程 (通常 是 第 一 个 ) 在 其 他 线程 运行 之 前 进入 confirmUpdate() 方 法 并 
取得 第 二 个 镇 。 这 种 情况 也 会 发 生 在 原始 代码 中 ， 只 是 可 能 性 更 低 罢了 。 


DD 自在 读书 @ 


Www .Zizidiary .com 


4.3 现代 并 发 应 用 程序 的 构件 。 85 


我 们 只 是 揭 开 了 Lock 各 种 可 能 性 的 面纱 一 一 有 很 多 种 方法 可 以 产生 更 加 复杂 的 锁定 结构 。 
接 下 来 我 们 就 来 讨论 其 中 一 个 概念 一 一 锁 存 知 。 


4.3.3 CountDownLatch 


countDownLatch 是 一 种 简单 的 同步 模式 , 这 种 模式 允许 线程 在 通过 同步 屏障 之 前 做 些 少量 
的 准备 工作 。 

为 了 达到 这 种 效果 ， 在 构建 新 的 CountDownLatch 实 例 时 要 给 它 提供 一 个 int 值 (计数 器 )。 
此 外 ， 还 有 两 个 用 来 控制 锁 存 器 的 方法 : countDown () 和 await ()。 前 者 对 计数 器 减 ]， 而 后 者 
让 调用 线程 在 计数 器 到 0 之 前 一 直 等 待 。 如 果 计数 器 已 经 为 0 或 更 小 , 则 它 什么 也 不 做 。 这 个 简单 
的 机 制 使 得 这 种 所 需 准 备 最 少 的 模式 非常 容易 部 署 。 

在 下 面 的 代码 中 , 同一 进程 内 的 一 组 更 新 处 理 线程 至 少 必 须 有 一 半 线 程 正确 初始 化 (假定 更 前 
处 理 线程 的 初始 化 要 占用 一 定时 间 ) 之 后 ,才能 开始 接受 系统 发 送 给 它们 中 的 任何 一 个 线程 的 更 新 。 


代码 清单 4.7 ”用 锁 存 器 辅助 初始 化 
public static class ProcessingThread extends Thread | 
private final string ident; 
private final CountDownLatch latch:; 


public ProcessingThread(lString ident , CountDownLatch cdl ) 1 
ident = ident ; 
latch = cdl ; 

| 

publiec String getIidentifier() | 
return identifier:; 


od | 节点 初始 化 
public woid initialize(}) 1 - 


latch .countDown'(): 
} 
public void run() 1 
initializel(l); 


| 
} 


final int quorum = 1 + (int) (MAX THREADS / 2); 
final CountDownLatch cdl] = new CountDownLatch (guorum); 


final Set<ProcessingThread> nodes = new HashSet<>s/(); 
try 1 
for lint i=0; i<MAX THREADS; i++) | 
ProceasingThread local = new ProcessingThread("localhost:"+ 
ss (9000 + 寺 ) ， cdl); 
nodes.add(local).; 


local .start {():; 达到 auorum, 开始 
cdl .await () : 寸 发 送 更 新 


} catch (InterruptedException e) 1 
} finally 
| 
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这 段 代 码 把 锁 存 器 的 值 设 置 为 quorum, 一 旦 被 初始 化 的 线程 达到 这 个 数量 ， 就 可 以 开始 处 
理 更 新 了 。 每 个 线程 完成 初始 化 后 都 会 马上 调用 countDown() ， 所 以 主线 程 只 需 等 待 auorum 的 
到 来 ， 然 后 启动 ( 并 派发 更 新 ， 尽 管 我 们 设 给 出 那 部 分 代码 )。 

我 们 接 下 来 要 讨论 的 是 对 多 线程 开发 人 员 来 说 最 有 用 的 类 之 一 : java.util.concurrent 


中 的 concurrentHashMap。 
4.3.4 ConcurrentHashMap 


concurrentHashMap 类 是 标准 HashMap 的 并 发 版 本 。 它 改进 了 Collections 类 中 提供 的 
synchronizedMap1() 功 能 ， 因 为 那些 方法 返回 的 集合 中 包含 的 锁 要 比 需 要 的 多 。 


"EAS I777 
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图 4-7 HashMap 的 经 典 视图 


如 图 4-7 所 示 ， 传 统 的 HashMap 用 hash 函 数 来 确定 存放 键 / 值 对 的 “ 桶 ”， 这 是 该 类 和 名字 中 
“Hash” 的 由 来 。 这 意味 着 多 线程 处 理 可 以 更 加 简单 直接 一 一 修改 HashMap 时 并 不 需要 把 整个 结 
构 都 锁 住 ， 只 要 锁 住 即将 修改 的 桶 就 行 了 。 


提示 “好 的 并 发 HashMap 实 现在 读 取 时 不 用 锁 , 写 入 时 只 需 锁 住 要 修改 的 桶 。Java 基 本 上 能 达到 
这 个 标准 ， 但 这 里 还 有 一 些 大 多 数 开 发 人 员 都 无 需 过 多 关注 的 底层 细节 。 


ConcurrentHashMap 类 还 实现 ConcurrentMap 接 口 ， 有 些 提 供 了 原子 操作 的 新 方法 。 

口 putIfAbsent () 一 一 如 末 还 没有 对 应 键 ， 则 把 键 / 值 对 添加 到 HashMap 中 。 

口 remove () 一 一 如 果 对 应 键 存在 , 且 值 也 与 当前 状态 相等 ( equal ), 则 用 原子 方式 移 除 键 

值 对 。 

口 replace() 一 一 API 为 HashMap 中 原子 替换 的 操作 方法 提供 了 两 种 不 同 的 形式 。 

比如 说 ， 如果 你 把 代码 清单 和 -1 中 的 私有 final 域 arrivalTime 的 类 型 从 HashMap 改 成 
concurrentHashMap， 那 就 可 以 把 synchronized 方 法 替换 成 常规 的 非 同 步 访问 。 注意 代码 清 
单 4-8 中 锁 的 缺失 一 一 根本 就 没有 显 式 的 同步 。 
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public class ExampleMicroBlogTimingNode implements SimpleMicroBlogNode | 


private final Map<Update, Long> arrivalTime = 
wk New ConcurrentHashMap <>().; 


public void propagateUpdate(Update upd ) | 
arrivalTime.putIfAbsent (upd , System.currentTimeMillis()); 


public boolean confirmUpdateReceived(lUpdate upd ) | 
return arrivalTime .get (upd ) != null, 
| 
} 


ConcurrentHashMap 是 java.util.concurrent 包 中 最 有 用 的 类 之 一 。 它 不 仅 提供 了 多 线 
程 的 安全 性 ， 并 且 性 能 更 优 ， 在 日 常 使 用 中 没有 严重 的 缺陷 。 接 下 来 我 们 会 讨论 它 的 最 佳 拍档 ， 
用 于 List 的 copyOnWriteArrayList。 

4.3.5 CopyOnWriteArrayList 

从 名 字 就 能 看 出 来 , CopyonWriteaArrayList 是 标准 ArrayList 的 替代 品 。 CopyOnWrite- 

ArrayList 通 过 增加 写 时 复制 ( copy-on-write ) 语义 来 实现 线程 安全 人 性， 也 就 是 说 修改 列表 的 任 


何 操作 都 会 创建 一 个 列表 底层 数组 的 新 复 本 ( 如 图 4-8 所 示 )。 这 就 意味 着 所 有 成 形 的 迭代 器 "都 
不 用 担心 它们 会 碰 到 意料 之 外 的 修改 。 


被 修改 的 数组 人 广 一 一 
上 的 选 代 器 | | ss 
物 . 而 
图 4-8 写 时 复制 数组 


当 快 速 、 一 致 的 数据 快照 (不同 的 读 取 些 读 到 的 数据 偶尔 可 能 会 不 一 样 ) 比 完 美的 同步 以 及 
性 能 上 的 突破 更 重要 时 ， 这 种 共 至 数据 的 方法 非常 理想 ， 并 经 党 出 现在 非 天 键 任务 中 。 

我 们 来 看 一 个 写 时 复制 的 案例 。 假设 有 个 微 博 的 时 间 线 更 新 , 这 是 一 个 典型 的 非 关键 任务 的 
例子 。 每 个 读 取 器 的 性 能 、 目 映 一 致 性 的 快照 要 比 全 局 的 一 致 性 更 受 欢 迎 。 代 码 清 单 4-9 表 示 每 
个 用 户 时 间 线 视图 的 持 有 者 类 。 我 们 将 会 在 代码 清单 4-10 中 用 它 来 演示 写 时 复制 操作 是 如 何 进 
行 的 。 


DD 迁 代 器 ( iterator ) 是 一 个 对 象 ， 它 的 工作 是 遍历 并 选择 序列 中 的 对 象 ， 而 客户 端 程序 员 不 必 知 道 或 美 心 该 序列 底 
层 的 结构 (也 就 是 不 同 容 器 的 类 型 )。 一 一 详 者 注 
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public class MicroBlogTimeline | 
private final CopyOnWritehrrayList<Update> Updates:; 
private final ReentrantLock lock.; 
private final String name:; 


private Iterator<Update> 1t; < 
public void addUpdate (Update update ) 1 构造 方法 已 省 略 


updates.add (update }; 


} 设置 选 代 器 
oublic void prep() 1 
it = Updates.iterator'(); 
} 
public void printTimeline() 1 
lock, lock'():; 三 - 下 本 
ery | | 需要 在 这 里 锁定 


if (it l= null) 1 
System.out .print (name+ ™: "),) 
while (it.hasNext ()) 1 
Update 8 = it .nextl()}; 
Syatem.out .print (s+ ", "); 
| 
System.out .println(}; 
| 
} finally { 
Lock .unlock!(}; 
} 
和 
} 
我 们 专门 设计 了 这 个 类 来 阐明 在 写 时 复制 语义 下 的 过 代 带 行 为 。 你 需要 在 输出 方法 中 锁定 ， 
以 防止 输出 在 两 个 线程 间 乱 掉 ， 此 外 你 也 能 看 到 两 个 线程 各 自 的 状态 。 


你 可 以 从 下 面 的 代码 中 调用 MicroBlogTimeline 类 。 
代码 清单 4-10 ”揭示 写 时 复制 行为 


final CountDownLatch firstLatch = new CountDownDatcech (1) ; 
final CountDownLatch secondLatch = new CountDownLatcht(1)}., 
final Update.Builder ub = new Update .Builder 1 1) ， 


了 设置 初始 状态 


final List<Updates 1 = new CopyOnWriteArrayLiast<>(); 
l .add(ub,.author (new Author ("Ben")) updateText ("I like pie") .buildt{))., 
l.addlub.author (new Author ("Charles")) updateText I 

my "TIT like ham on rye") .build'())}).; 


ReentrantLock lock = new ReentrantLockl().; 
final MicroBlogTimeline tl1 = new MicroBlogTimeline ("TL1", 1, lock).; 
final MicroBlogTimeline tl12 = new MicroBlogTimeline("TL2", 1, lock)'; 


Thread tl = new Threadf) f 
public void run{(}) 1 
ll .add (ub.author (new Author ("Jeffrey")) .updateText ( 
= "I like a lot of thingas") .buila()).; 
tl11 .prep(): 
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firatLatceh.countDown(); 
try { secondLatch.await(}); } 
mw» catch (InterruptedException e) { |} 
tl1l1.printTimeline(); 
} 
}; 
Thread t2 = new Thread()1 
public void run1)1{ 
try 1 
firastLatch.await(),; 
l1.add(ub.author(lnew Author("Gavin")) .updateText | 
mk "I like otters") .build(})}); 
tl2.prepl})? 
| 


secondLatch.countDown |(); 本 
} catch (InterruptedException el ( | 用 和 锁 存 器 严格 限制 
tl2.printTimeline(); 事件 的 顺序 
}; 


tl1.start'(); 
t2 .start(}; 


这 段 代码 里 有 很 多 辅助 的 测试 代码 。 但 也 有 很 多 值得 注意 的 地 方 : 

Count DownLatch 用 来 严格 控制 两 个 线程 之 间 发 生 的 事情 。 

口 如 果 用 普通 的 Li st 代替 CopyOnWwri teArrayList, 结果 会 导致 出 现 ConcurrentModi- 

ficationException 异 常 。 

口 这 也 是 在 两 个 线程 之 间 共 享 一 个 Lock 对 象 以 控制 对 共享 资源 ( 即 sTDoUT ) 访问 的 例子 。 

如 果 用 块 结构 方 式 写 这 段 代码 ， 会 显得 更 加 杂乱 。 

这 段 代 人 码 的 输出 如 下 : 

TL2: Update [author=Author [name=Ben], updateText=I like pie, createTime=0], 
Update [author=MAuthor [name=Charlesa], updateText=I like ham on rye, 
createTime=0], Update [author=Author [name=Jeffrey], updateTexta=l like a 
lot of things, createTime=0], Update [author=Author [name=sGavin] ， 
updateText=I like otters, createTime=0], 


用 锁 存 器 严格 限制 
事件 的 顺序 


TL1: Update [lauthor=Author [name=Ben], updateText=I like pie, createTime=0], 
Update [author=Author [name=Charles], updateText=I like ham on rye, 
createTime=0], Update [author=Author [name=Jeffrey], updateTexta=Il like a 
lot of things, createTime=0], 
第 二 行 输出 ( 标签 为 TL1 ) 漏 掉 了 最 后 一 个 更 新 ， 就 是 提 到 了 水 猫 的 那个 ， 尽 管 按 锁 存 器 的 
意思 在 列表 被 修改 后 L110 是 可 以 访问 的 。 这 说 明了 t11 中 所 包含 的 迭代 器 被 -12 复 制 ， 并 且 最 后 
一 个 更 新 对 t11 是 不 可 见 的 。 这 就 是 我 们 想 要 展示 的 写 时 复制 特性 。 


RS 


HashMap 的 即 用 型 并 发 替代 品 -这 是 因为 性 能 问题 一 写 时 复制 特性 意味 着 如 果 列 表 在 被 读 取 


山 原文 为 mbexl1， 下 文 同 。 一 一 详 者 注 


只 自在 读书 @3 


www .zizidiary .com 


90 第 4 章 现代 并 发 


或 遍历 时 做 了 修改 ， 那 就 必须 复制 整个 数组 。 

也 就 是 说 如 果 对 列表 的 修改 次 数 跟 读 取 次 数 相差 不 多 ， 这 种 方式 未 必 能 达到 较 好 的 性 能 。 
但 就 像 我 们 在 第 6 章 一 再 提 到 的 那样 ， 得 到 性 能 优异 的 代码 的 唯一 可 靠 的 方法 就 是 测试 ， 再 测 
试 ， 并 衡量 结果 。 


下 一 个 在 并 发 代码 中 常用 的 构件 是 java .util.concurrent 中 的 oueue。 它 用 于 在 线程 之 
间 切 换 工 作 元 素 ， 并 且 还 是 很 多 灵活 可 徘 的 多 线程 设计 的 基础 。 


4.3.6 Queue 


队列 是 一 个 非常 美妙 的 抽象 概念 。 不 , 之 所 以 这 么 说 并 不 是 因为 我 们 生活 在 伦敦 这 个 世界 排 
队 之 都 。 为 把 处 理 资源 分 发 给 工作 单位 【或 者 把 工作 单元 分 配给 处 理 资 源 ,， 这 取决 于 你 看 待 问题 
的 方式 )， 队 列 提供 了 一 种 简单 又 可 靠 的 方式 。 

Java 中 有 些 多 线程 编程 模式 在 很 大 程度 上 都 依赖 于 Queue 实 现 的 线程 安全 性 ， 所 以 很 有 必要 
充分 认识 它 。oueue 接 口 被 放 在 了 java.utili 包 中 ,因为 即便 在 单线 程 编 程 中 它 也 是 一 个 重要 的 
模式 ， 但 我 们 的 重点 是 多 线程 编程 ， 并 且 假 年 你 已 经 熟悉 队列 的 基本 用 法 了 。 

队列 经 笛 用 来 在 线程 之 间 传 递 工 作 单 元 ， 这 个 模式 通 汕 适合 用 Queue 最 简单 的 并 发 扩展 
Blockingoueue 来 实现 。 接 下 来 我 们 就 会 重点 介绍 它 。 

1. BLockingoueue 

BlockingQueue 还 有 两 个 特性 。 

口 在 向 队列 中 put () 时 ， 如 果 队 列 已 满 ， 它 会 让 放 人 线程 等 待 队 列 腾 出 空间 。 

口 在 从 队列 中 take() 时 ， 如 果 队 列 为 宝 ， 会 守 致 取出 线程 阻 宕 。 

这 两 个 特性 非常 有 用 ， 因 为 如 果 一 个 线程 ( 或 线程 池 ) 的 能 力 超过 了 其 他 线程 ， 比 较 快 的 线 
程 就 会 被 强制 等 待 ， 因 此 可 以 对 整个 系统 起 到 调节 作用 ， 如 图 4-9 所 示 。 


A 必须 等 待 
琶 程 昌 
take (外 
队列 空 [2 一。 
8 必须 等 待 : 


图 4-0 BlockingQueue 
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”Java 提 供 了 BlockingQueue 接口 的 两 个 基本 实现 : LinkedBlockingQueue 和 
ArrayBlockingQueue。 它 们 的 特性 稍 有 不 同 ; 比如 说 ， 在 已 知 队列 的 大 小 而 能 确定 合适 的 
边界 时 ， 用 ArrayBlockingQueue 非 常 高 效 ， 而 LinkedBlockingQueue 在 某 些 情况 下 则 会 
快 一 点 儿 。 


2. 使 用 工作 单元 

Queue 接 口 全 都 是 泛 型 的 一 一 它们 是 Queue<E>，BlockingQueue<EB>， 等 等 依 此 类 推 。 尽 
管 看 起 来 奇怪 ， 但 有 时 候 利 用 这 一 点 把 工作 项 封装 在 一 个 人 工 容 器 类 内 却 是 明智 之 举 。 

比如 说 ， 你 有 一 个 表示 工作 单元 的 MyvaAwesomeclass 类 ， 想 要 用 多 线程 方式 处 理 ， 与 其 用 
Blockingoueue<MyawesomeClass> 不 如 用 B1 CCKingOueue<WorKkUniL<MYawWeScCrmeC1aSS>>n 
其 中 WorkUnit( 或 ueueObject，, 或 随 你 怎么 命名 这 个 容器 类 ) 是 像 下 面 这 样 的 包装 接口 或 类 ， 


public class WorkUnit<T> 1 
private final T workUnit., 


public T getWork{){| return workUnit; } 


public WorkUnit (T workUnit ) | 
workUnit = workUnit ; 
} 
} 


有 了 这 层 间 接 引 用 ， 不 用 牺牲 所 包含 类 型 ( 在 此 即 MyAwesomeclass ) 在 概念 上 的 完整 性 就 
可 以 在 这 里 添加 额外 的 元 数据 了 。 

这 特别 有 用 。 人 能 用 上 额外 元 数据 的 用 例 很 多 ， 下 面 举 几 个 例子 : 

口 测试 ( 比如 展示 一 个 对 象 的 修改 历史 ) 

口 性 能 指标 ( 比如 到 达 时 间或 服务 质量 ) 

口 运行 时 系统 信息 ( 比如 MyAwesomeclass 实 例 是 如 何 被 排 到 队列 中 的 ) 

以 后 再 在 这 种 间接 引用 里 增加 元 数据 可 能 会 非常 困难 。 如 果 你 发 现在 某 些 情况 下 需要 更 多 的 
元 数据 ， 那 么 要 把 它们 加 和 到 间接 引用 中 可 能 需要 大 量 的 重 构 工 作 ， 而 加 在 workUnit 类 中 就 只 
是 个 简单 的 修改 。 

3. 一 个 BlockingQueue 的 例子 

我 们 用 一 个 简单 的 例子 - 年 着 看 医生 的 宠物 们 一 一 来 看 看 如 何 使 用 Blockingoueue。 这 
个 例子 中 有 一 个 等 着 让 医生 给 做 检查 的 宠物 集合 。 


代码 清单 4-11 在 Java 中 对 宠物 建 模 
public abstract class Pet | 
protected final String name: 


public Pet (String name) { 
this.name = Name; 


| 
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public abstract void examine()} 


| 
public class Cat extends Pet | 
public Cat (string name) | 
super (name); 


} 
public void examine() |{ 
System.out .println("Meow!"); 
| 
| 
public tlass Dog extends Pet 
publie Dogl!string name) 1 
SUpPer (name); 
} 
public void examine(} | 
Svystem.Gut .printl]n("Wootf!"); 
| 
| 


public class Appointment<T> { 
private final T toBeSeen,; 
public T getPatient () | return tobBeSeen:; ] 


public Appointment (T incoming) { 
toBeSeen = incoming; 
| 


在 这 个 简单 的 例子 中 , 我 们 用 LinkedBlockingQueue<Appointment<Pe t>> 表 示 兽 医 的 候 
诊 队 列 Appointment 起 到 ] WorkUnit 的 作用 。 

兽医 对 象 是 由 一 个 队列 和 一 个 暂停 时 间 构 建 的 , 其 中 队列 是 由 一 个 代表 接待 员 的 对 象 提供 的 
预约 队列 ， 和 暂停 时 间 表 示 曾 医 在 预约 之 间 的 停工 时 间 。 

我 们 可 以 在 下 面 这 段 代码 中 建立 兽医 的 模型 。 在 线程 运行 时 , 它 在 一 个 无 限 循 环 中 重复 调用 
seePatient ()。 当 然 ， 现实 世界 中 的 兽医 不 可 能 这 样 ， 因 为 他 们 晚上 和 周末 要 回 家 ,不 能 一 和 直 
在 办 公 室 等 着 生病 的 小 动物 上 | ] 就 医 。 


代码 清单 4-12 对 兽 十 人 . 


public class Veterinarian extends Thread | 
protected final BlockingQueue<Appointment<Pet>> appts; 
protected String text = "",; 
protected final jint reatTime; 
private boolean shutdown = false; 


public Veterinarian(Blockingoueue<Appointment<Pet>»> lbq, int pause) I 
appts = lba:; 
restTime = pause; 

| 

public synchronized void shutdown()1 
shutdown = true; 


| 
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EOverride 
public void run(})l| 
while (lghutdown) | 
seePatient!():; 
try | 
Thread.sleep [restTime).; 
] catch (InterruptedException e) 1 
shutdown = true; 
| 
| 
| 


public void seePatient {) | 


| | 阻塞 cake 
Appointment<Pet> ap = apptSs.takel(1) ; - 


Pet patient = ap.getPatient (); 
patient .examine|(),; 
] catch (InterruptedException e€) | 
shutdown = true,; 
} 
| 
} 
在 seePatient |) 方 法 中 , 线程 会 从 队列 中 取出 预约 ， 并 挨个 检查 对 应 的 宠物 ， 如果 当前 队 
列 中 没有 预约 等 待 ， 则 会 阻塞 。 
4. BLockingoueue 的 细 和 粒度 控制 
除了 简单 的 Lake () 和 of fer()API, Blockingoueue 还 提供 了 另外 一 种 与 队列 交互 的 方式 ， 
这 种 方式 对 队列 的 控制 力度 更 大 , 但 稍微 有 点 复杂 。 这 就 是 带 有 超时 的 放 入 或 取出 操作 , 它 允 许 
线程 在 过 到 问题 时 可 以 从 与 队列 的 交互 中 退出 来 ， 转 而 做 点 儿 其 他 的 事情 。 
实际 上 ， 这 个 功能 并 不 常用 , 但 它 偶 尔 能 派 上 大 用 场 ， 所 以 我 们 要 介绍 一 下 。 下 面 的 例子 还 
是 来 目 微 博 。 


代码 清单 4-13 ”BlockingQueue 行 为 的 例子 


public abstract class MicroBlogExampleThread extends Thread | 
protected final BlockingQueue<Update> Updates:; 
protected String text = ""; 
protected final int pauseTime; 
private boolean shutdown = false; 


public MicroBlogExampleThread(lBlockingQueue<Update>s 1lbq , int pause ) | 
Updates = lbqg i; 


pauBeTime = pause ;} 

} 

public synchronized void shutdown()| 
shutdown = true:; 


| 
Override 
Public void runf){ 
while (i'shutdown) | 9 使 线程 可 以 彻底 地 结束 


doAction(}:; 
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Cry 1 
Thread.sleep lpauseT1ime); 
} catch (InterruptedException e) 1 
shutdown = true; 
} 
| 


更 0 
public absatract void doAction|); i 由 子 类 实现 具体 动作 


| 
final Update.Builder ub = new Update.BulLader11 : 
final BlockingQueue<Update>s 1bq = new LinkedBlockingQueue<> (100}).; 
MicroBlogExampleThread t1 = new MicroBlogExampleThread(lbg, 10) | 
public void doAction() 1 
text = text + "XK" 
Update ui ss Ub.author (new Author ("Tallulah")}) .updateText (text) .buildl(),; 
boolean handed = false; 
try 1 
handed = updates.coffer(u, 100, TimeUnit ,MILLISECONDS) )， 
} catch (InterruptedException e) 1 


if (!handed) System.cut .println'! 
mr "Unable to hand off Update to Queue due to timeout"); 


}7 
MicroBlogExampleThread t2 = new MicroBlogExampleThread(lbq, 1000) 1 
public void doAction(}) | 
Update uu = null; 
try 1 
u = Updates.take(},; 
} catch (InterruptedException e) | 
return) 
} 
} 
}; 
tl.start(}):; 
t2.8tart{); 
运行 这 段 代码 展示 了 填充 队列 的 速度 有 和 多么 快 , 也 表明 供给 线程 的 速度 超过 了 提取 线程 的 速 
度 。 很 快 , “Unable to hand off Update to Queue due to timeout” 消 息 就 出 现 了 。 
这 是 “相连 线程 池 ” 中 的 一 种 典型 的 极端 状况 ， 当 上 洲 的 线程 池 比 下 游 的 快 ， 这 种 情况 就 会 
发 生 。"“ 相 连 线程 池 ” 可 能 会 引发 一 些 问 题 ， 比 如 会 守 致 LinkedBlockingQueue 淤 出 。 男 外 ， 
如 采 消 费 者 比 生 产 者 多 ， 队 列 会 因此 而 人 经常 空 着 。 好 在 Java 7 在 Blockingoueue 上 有 了 解决 办 


TransferOQueue, 

5. TransferQueue 一 一 Java 7 中 的 新 贵 

Java 79| 人 J 了 TransferQueue。 它 本 质 上 是 多 了 一 项 transfer() 操 作 的 BlockingQueue。 
如 果 接 收 线程 处 于 等 待 状态 , 该 操作 会 马上 把 工作 项 传 给 它 。 否 则 就 会 阻塞 直到 取 走 工作 项 的 线 
程 出 现 。 你 可 以 把 这 看 做 “挂号 信 ” 选 项 ， 即 正在 处 理工 作 项 的 线程 在 交付 当前 工作 项 之 前 不 会 
开始 其 他 工作 项 的 处 理工 作 。 这 样 系统 就 可 以 调控 上 游 线程 池 获取 新 工作 项 的 速度 。 
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用 限定 大 小 的 阻塞 队列 也 能 达到 这 种 调控 效果 , 但 TransferQoueue 接 口 更 灵活 。 此 外 ， 用 
Transteroueue 取代 Blockingoueue 的 代码 性 能 表现 可 能 会 更 好 。 这 是 因为 编写 
TransferQueue 的 实现 时 已 经 将 现代 编译 右 和 人 处 理 颖 的 特性 考虑 在 内 ,执行 起 来 效率 更 高 。 聊 
了 这 人 么 久 性 能 ， 不 能 室 口 无 任 ， 必 须 给 出 测量 结果 并 能 证 明 才 行 。 另 外 你 也 应 该 意识 到 ，Java 7 
只 给 出 了 了 Transferoueue 的 一 种 实现 形式 一 一 链表 版 。 

在 下 面 的 例子 中 ， 你 会 发 现 用 TransferQoueue 代 替 BlockingQueue 是 多 么 简单 。 只 要 对 清 
单 4-13 中 的 代码 做 些 简单 修改 ， 就 可 以 升级 成 Transferoueue， 请 看 这 


代码 清单 4-14 用 Transferoueue 代 替 Blockingoueue 
public abstract class MicroBlogBxampleThread extends Thread | 
protected final TransferQueue<Update> updates;} 


public MicroBlogExampleThread(TransferQueue<Update> lbqg ，int pause ) | 
updates = lbqg ， 
pauseTime = pause | 
} 
} 
final TransferQueue<Update> lbq = new LinkedTransferQueue<Update>(100); 
MicroBlogExampleThread tl1 = new MicroBlogExampleThread(lbg, 10) 1 
public void doAction|)1 


try | 
handed = updates.tryTransfer(u, 100, TimeUnit .MILLISECONDS):; 
} catch (InterruptedException e) 1 


| 
} 
1 
到 此 为 止 , 用 来 开发 多 线程 应 用 的 主要 构件 我 们 都 见识 过 了 。 接 下 来 该 把 它们 整合 到 驱动 并 
发 代码 的 引擎 (执行 器 框架 ) 上 了 。 用 它们 可 以 对 任务 进行 调度 和 控制 ,可 以 组 合 高 效 的 并 发 流 
处 理工 作 项 ， 从 而 构建 大 型 多 线程 应 用 程序 。 


4.4 控制 执行 


我 们 在 前 面 的 讨论 中 一 直 把 工作 任务 当成 抽象 的 单元 。 然 而 有 个 细节 需要 注意 , 我 们 一 直 没 
有 提 到 的 是 这 些 单元 要 比 Thread 小 一 一 它们 提供 的 方法 把 计算 任务 包含 在 一 个 工作 单元 中 , 无 
需 为 每 个 单元 局 动 新 的 线程 。 这 样 处 理 多 线程 代码 通常 效率 更 高 ， 因 为 免除 了 为 每 个 单元 启动 
Thread 的 开销 。 执 行 代 码 的 线程 是 重用 的 ， 处 理 完 一 个 任务 后 会 继续 处 理 新 的 工作 单元 。 

虽然 复杂 一 些 , 但 你 可 以 实现 线程 池 、 工人 与 管理 者 模式 和 执行 者 等 开发 人 员 最 常用 的 模式 。 
我 们 接 下 来 要 密切 关注 可 以 对 任务 (callable、Future 和 FutureTask ) 和 执行 者 建 模 的 类 和 
接口 ， 特 别 是 ScheduledThreadPoo1Executor。 
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4.4.1 任务 建 模 


我 们 的 终极 目标 是 不 用 为 调度 每 个 任务 或 工作 单元 而 局 动 新 线程 。 归根 结 展 ， 融 是 要 把 它们 
做 成 可 以 调用 (通常 由 执行 者 调用 ) 的 代码 ， 而 不 是 直接 可 运行 的 线程 。 

我 们 来 看 对 任务 建 模 的 三 种 办 法 一 一 callable 和 Future 接 口 以 及 FutureTask 类 。 

1. callable 接 口 

Callable 接 口 是 一 个 非常 常见 的 概念 ， 代 表 了 一 段 可 以 调用 并 返回 结果 的 代码 。 尽 管 这 种 
做 法 很 直接 ， 但 实际 上 它 的 作用 微妙 而 又 强大 ， 用 它 可 以 创建 出 一 些 特别 实用 的 模式 。 

callable 的 典型 用 法 是 匿名 实现 类 。 这 段 代 人 码 的 最 后 一 行 把 s 赋 值 为 out .toString(): 


final MyObject out = getSampleOQbject():; z 
callable<String> cb = new Callable<Strings() | 
public String call() throws Exception | 
return out.tostring(}); 
} 

上 
string 8 = cb.call').; 

可 以 把 callable 的 匿名 实现 类 当做 对 单一 抽象 方法 call () 的 化 延 调用 , 该 实现 必须 提供 这 
个 方法 。 

callable 是 SAM 类 型 (“单一 抽象 方法 ”的 纺 号 ， 有 时 会 这 样 称呼 它 ) 的 示例 一 一 这 是 Java 
7 把 函数 作为 一 等 类 型 最 可 行 的 办 法 。 在 后 续 章 节 讨 论 非 Java 语 言 时 还 会 过 到 它们 ， 那 时 我 们 还 
会 进一步 讨论 把 函数 作为 值 或 一 等 类 型 的 概念 。 

2. Future 接 口 
接口 用 来 表示 异步 任务 ， 是 还 没有 完成 的 任务 给 出 的 未 来 结果 。 我 们 在 第 2 章 介 绍 
NIO.2 和 异步 WO 时 提 过 ，。 

下 面 是 Future 中 的 主要 方法 。 

口 get () 一 一 用 来 获取 结果 。 如 果 结 果 还 设 准备 好 ，get () 会 被 阻塞 直到 它 能 取得 结果 。i 

有 一 个 可 以 设置 超时 的 版 本 ， 这 个 版 本 永远 不 会 阻塞 

D cancel () 一 一 在 运算 结束 前 取消 。 

口 ispone() 一 一 调用 者 用 它 来 判断 运算 是 否 结束 。 

下 面 这 段 代码 ( 找 素 数 ) 展示 了 Future 的 用 法 : 


Future<Longs> fut = getNthPrime (1 000 000 000); 


Long reault = null, 
while (result == null) 1 
try 1 
result = fut.get (60, TimeUnit .SECONDS) ; 
} catch (TimeoutException tox) { } 
system.out .println("still not found the billionth prime!"):; 
| 


System.out printlnl"Found it: "* result.longValue(})),) 
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在 这 段 代码 中 ， 你 应 该 想象 一 下 返回 Future 的 getNthPrime() 在 某 个 后 人 台 线 程 或 多 个 线程 
上 运行 的 情景 , 也 有 可 能 是 在 执行 者 框架 上 和 运行。 即便 使 用 先进 的 硬件 ,这 种 运算 可 能 也 需要 很 
长 时 间 你 最 后 还 是 要 用 Future 的 cancel |) 方 法 。 

3. FutureTask 类 

FutureTask 古 Future 接 口 的 常用 实现 类 ， 它 也 实现 乓 Runnable 接 口 。 这 意味 着 
FutureTask 可 以 由 执行 者 调度 ， 这 一 点 很 关键 。 它 对 外 提供 的 方法 基本 上 就 是 Future 和 
Runnable 接 口 的 组 合 : get () 、cancel()、isDone() 、isCancelled() 和 run|()， 最 后 一 个 
方法 通常 都 是 由 执行 者 调用 ， 你 基本 不 需要 直接 调用 它 。 

FutureTaskx 还 提供 了 两 个 很 方便 的 构造 器 : 一 个 以 Callable 为 参数 ， 另 一 个 以 Runnable 
为 参数 。 这 些 类 之 间 的 关联 表明 对 于 任务 建 模 的 办 法 非常 灵活 ， 人 允许 你 基于 FutureTask 的 
Runnable 特 性 ( 因为 它 实 现 了 Runnable 接 口 )， 把 任务 写成 callable， 然 后 封装 进 一 个 由 热 
行者 调度 并 在 必要 时 可 以 取消 的 FutureTask。 


4.4.2 ScheduledThreadPoolExecutor 


scheduledThreadPoolExecutor (以 下 简称 STPE ) 是 线程 池 类 中 的 重 中 之 重 一 一 它 功 能 
多 样 ， 广 释 欢 迎 。STPE 接 收 任务 ， 并 把 它们 安排 给 线程 池 里 的 线程 。 

口 线程 池 的 大 小 可 以 预定 义 ， 也 可 自 适 应 。 

口 所 安排 的 任务 可 以 定期 执行 ， 也 可 只 运行 一 次 。 

口 STPE 扩 展 了 ThreadPoolExecutor 类 (很 相似 , 但 不 具备 定期 调度 能 力 )。 

和 java.util.concurrent 中 的 工具 类 相 结 合 的 STPE 线 程 池 是 大 中 型 多 线程 应 用 程序 最 
常见 的 模式 之 一 ， 这 些 工 具 类 包括 我 们 在 前 面 已 经 见 过 的 Concnu rrentHashMap, CopyoOnWrite- 
ArrayList 和 BlockingQueue 等 。 

STPE 不 过 是 通过 Executors 类 的 工厂 方法 轻易 获取 的 众多 执行 者 之 一 。 使 用 这 些 工厂 方法 
很 方便 ， 开 发 人 员 通 过 它们 可 以 轻易 获取 典型 配置 ,需要 时 还 可 以 开放 完整 的 接口 方法 。 

下 面 的 代码 是 一 个 定期 访 取 的 例子 。 这 是 newscheduledThreadPool1() 的 常见 用 法 ; 
msgReader 对 和 象 被 安排 pol1 () 一 个 队列 ， 从 队列 中 的 workUnit 对 象 里 取得 工作 项 ， 然 后 输出 。 


private ScheduledExecutorService stpe:; | 取消 时 需要 


private ScheduledFuture<?> hndl.:; 


private BlockingQueue<WorkUnit<String>> 1bqg = new LinkedBlockingOueue<>|(); 
private void run()1 
stpe = Executors.newScheduledThreadPool (2):; \ ] 
”| 执行 者 的 工厂 方法 
final Runnable msgReader = new Runnable()1 
public void run1) { 
String nextMsg = lbg.poll{() .getWork1); 
if {nextMsg l= null} System.out .println("Msg racvd: "+ nextMag}, 
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| 

}; 

nndl = stpe.scheduleAtFixedRate (magReader, 10, 10, 
mp TimeUnit .MILLISECONDS) ; 

| 

public void cancel() | 
final ScheduledFuture<?> myHndl = hnal:; 


stpe.schedule (new Runnable() | 电 :,H | 
了 er | ec NN } Wo EN 
}, 10, TimeUnit .MILLISECONDS) ; 
} 
在 这 个 例子 中 ，STPE 每 隔 10 毫 秒 就 唤醒 一 个 线程 ， 让 它 尝 试 poll () 一 个 队列 。 如 果 读 取 返 
回 null (因为 队列 当前 为 室 )， 则 什么 也 不 会 发 生 ， 线 程 回 去 继续 睡 大 觉 。 如 果 收 到 了 一 个 工作 
单元 ， 则 线程 会 输出 该 工作 单元 的 内 容 。 
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形 lable、FutureTask 及 相关 类 存在 几 个 问题 -一 尤其 在 涉及 类 型 系统 时 ，。 

要 明白 这 一 点 ,可 以 想 想 怎么 才能 满足 一 个 未 知 方法 可 能 出 现 的 所 有 方法 签名 -Callable 
只 能 用 于 没有 参数 的 方法 。 要 满足 所 有 可 能 性 ， 你 需要 Callable 的 不 同 变 体 。 

在 Java 中 ,你 可 以 通过 指定 模型 系统 内 的 方法 签名 来 解决 这 个 问题 ,但 你 在 本 书 第 三 部 分 
会 见 到 , 动态 语言 不 能 用 这 种 静态 视图 来 约 来 。 我 们 将 会 返回 来 重点 讨论 这 种 类 型 系统 之 间 的 
不 匹配 。 现 在 你 只 要 注意 到 ， 虽 然 Callable 很 有 用 ， 但 要 用 它 构 建 一 个 通用 框架 来 对 线程 执 
行进 行 建 模 还 是 有 点 儿 限 制 得 太 死 了 。 


现在 我 们 要 转向 Java 7 重点 突出 的 框架 之 一 一 一 用 于 轻 量 级 并 发 的 分 支 / 合 并 ( fork/join ) 框 
染 , 这 个 框架 比 我 们 在 本 节 中 见 到 的 执行 者 在 处 理 并 发 问题 方面 更 加 高 效 , 要 达到 这 点 绝 非 易 事 。 


4.5 分支 /合并 框架 


就 像 第 6 音 要 讨论 的 一 样 ， 处 理 器 的 速度 ( 或 者 更 准确 地 说 是 CPU 上 的 品 体 管 数量 ) 最 近 几 
年 增长 迅猛 。 由 此 产生 的 副作用 就 是 处 理 天 等 待 IO 操 作 变 成 了 家 和 常 便 饭 。 这 表明 我 们 能 够 更 好 
地 利用 计算 机 的 处 理 能 力 。 分 支 /合并 框架 就 可 以 解决 这 个 问题 一 一 这 也 是 Java 7 对 并 发 领域 新 做 
出 的 最 大 页 献 。 

分 支 / 合 并 框架 完全 是 为 了 实现 线程 池 中 任务 的 自动 调度 ， 并 且 这 种 调度 对 用 户 来 说 是 透明 
的 。 为 了 达到 这 种 效果 ， 必 须 按 用 户 指定 的 方式 对 任务 进行 分 解 。 在 很 多 应 用 程序 中 ， 对 于 分 支 
/合并 框架 来 说 都 可 以 很 自然 地 把 其 中 的 任务 分 成 “小 型 ”和 “大 型 ”任务 。 

我 们 来 快速 浏览 一 些 与 分 支 /合并 框架 相关 的 重要 事实 和 基本 原理 。 

口 引入 了 一 种 新 的 执行 者 服务 ， 称 为 ForkJoinPool。 

口 ForkJoinPool 服 务 处 理 一 种 比 线程 “更 小 ”的 并 发 单元 ForkJoinTask。 
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口 ForkJoinTask 是 一 种 由 ForkJoinPool 以 更 轻 量化 的 方式 所 调度 的 抽象 。 
口 通常 使 用 两 种 任务 ( 尽管 两 种 都 表示 为 ForkJoinTask 实 例 )， 
@ “小 型 ”任务 是 那 种 无 需 处 理 器 耗费 太 多 时 间 就 可 以 直接 执行 的 任务 。 
“大 型 ”任务 是 那 种 需要 在 直接 执行 前 进行 分 解 ( 还 可 能 不 止 一 次 ) 的 任务 。 
口 提供 了 支持 大 型 任务 分 解 的 基本 方法 ， 它 还 有 自动 调度 和 重新 调度 的 能 力 。 
这 个 框架 的 关键 特性 之 一 就 是 这 些 轻 量 的 任务 都 能 够 生成 新 的 ForkJoinTask 实 例 , 而 这 些 
实例 将 仍 由 执行 它们 父 任务 的 线程 池 来 安排 调度 。 这 就 是 分 而 治之 。 
我 们 会 通过 一 个 简单 的 例子 告诉 你 如 何 使 用 分 支 /合并 框架 ,然后 简短 介绍 “工作 窗 取 ”这 
个 特性 ， 最 后 讨论 一 下 那些 适用 于 并 行 处 理 技术 的 特性 。 使 用 分 支 /合并 框架 最 好 的 办 法 就 是 从 
例子 人 手 。 


4.5.1 一 个 简单 的 分 支 /合并 例子 


我 们 为 说 明 分 支 /合并 框架 而 设 定 了 这 样 的 应 用 场景 : 有 一 个 数组 ， 里 面 存 放 不 同时 间 到 达 
的 徽 博 更 新 ， 我 们 想 按 到 达 时 间 对 它们 排序 ， 以 便 为 用 户 生 成 时 间 线 ， 就 像 在 代码 清单 4-9 中 生 
成 的 那个 一 样 。 

我 们 会 用 Mergesort 的 变 体 实现 多 线程 排序 。 代 码 清单 4-16 中 用 到 了 ForkJoinTask 的 特定 
子 类 Recursiveaction。 因为 它 明 显 可 以 独立 完成 任务 ( 对 这 些 更 新 的 排序 能 当即 完成 ), 而 且 
具备 递归 处 理 能 力 (递归 特别 适合 做 排序 )， 所 以 用 Recursiveaction 会 比 用 通用 的 
ForkJoinTask 更 简单 。 

MicroBlogUpdateSorter 类 用 Update 对 象 的 compareTo() 方 法 对 更 新 列表 排序 。 
compute() 方 法 ( 超 类 RecursiveAction 中 的 抽象 方法 ， 必 须 实现 ) 基本 上 是 按 创建 时 间 对 微 
博 更 新 数组 排序 。 


代码 清单 4-16 ”用 RecursiveaAction 排 序 
public class MicroBlogUpdateSorter extende RecursiveAction | 
private static final int SMALL ENOUGH = 32; - 本 
private final Updatel] updates; 捉 行 排序 项 只 
private final int start, end) 有 32 个 或 更 少 
private final Update[] result:; 
public MicroBlogUpdateSorter (Update[] updates ) | 
this(updates , 0, updates .length):; 


Public MicroBlogUpdateSorter (Update[] upds ， 
“ int startPos , int endPos ) | 

start = StartPos ; 

end = endPos ; 

updates = Upds ; 

resgsult = new Update [updates. lengthl],; 


| 


private wvoid merge [MicroBlogUpdateSorter left ， 
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=“ MicroBlogqUpdateSorter right ) | 
int 1 = 0; 
int 1lCt = 0; 
int Tet ss OQ 
while (lcCt < left .size{() && rcCt < right .size()) { 
result [i++] = [left .result [lcCt] .compareTo lright .result [rcCt}}) < 0) 
? left .result[lCt++] 
right .result lrct++|]} 
} 
while (ct < left .size(}}) resuyult [i++] = left .result [lcCt++}); 
while (rcCt < right .size(})} result [i++ = right .result [rcCt++]; 


} 


Publie int size() 1 
return end - gtart; 


| 


public Update[] getResult() { 
return regsult.:; 


} 
二 RecursiveAction 
EOverride - 中 声明 的 方法 


protected void compute() 1{ 
it {size() < SMALL ENOUGH) 1 
Svatem.arraycecopy (Updates, start, result, 0, sizel)), 
Arrays .sort (result, 0, size()); 
} else | 
int mid = size() / 2; 
MicroBlogUpdateSorter left = new MicroBlogUpdateSorter | 
mb Updates, start, start + mid); 
MicroBlogUpdateSorter right = new MicroBlogUpdateSorter | 
mk. Updates, start + mid, end), 
invokeAll (left, right}); 
merge (left, right) 
| 
} 
} 


要 使 用 这 个 排序 器 ， 你 可 以 用 下 面 这 样 的 代码 驱动 它 ， 生 成 一 些 包含 由 X 组 成 的 字符 串 的 更 
新 ， 并 打 乱 它们 的 顺序 ， 之 后 青 传 给 排序 器 。 最 终 得 到 重新 排序 后 的 更 新 。 


Liast<Updaate> lu = new ArrayList<Update>!(); 
String text ss ™"; 

final Update.Builder ub = new Update.Builder|(); 
final Author a = new Authorl"Tallulah"),; 


for (int i=0; ic256; i*+) | 
text 三 text + “其 "; 
long now = System.currentTimeMillis'(); 
lu.add lub.author al .updateText (text) .createTime (now) .build'(})}); 
try 1 
Thread.sleep |1),} 


只 自 在 读书 @3 
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} catch [InterruptedException e) 1 
» 传 入 空 数组 ， 
collections.shuffle(lu); 省 掉 空间 分 配 


Update[] updates = lu.toArray (new Update[0]}., 


MicroBlogUpdateSorter sorter = new MicroBlogUpdateSorter (updates):; 
ForkJoinPool pool = new ForkJoinPool (4); 
pool .invoke (sorter); 


for {Update u: sorter.gatResult()}) 1 
System.out .println ilu),; 
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随 着 Java 7 的 到 来 ， pp i 但 到 了 
Java 本 变 成 了 “TimSort” 一 MergeSort pp 混合 体 , TimSort 最 初 是 Tim Peters 


pr pp 
组 。 对 数组 排序 时 ,由 于 数组 尺寸 太 小 , 会 调用 Array .sort () 方 法 ,该 方法 会 抛 出 空 指针 异 
常 ， 在 输出 的 异常 信息 里 就 能 看 到 Timsort 类 。 


4.5.2 ForkJoinTask 与 工作 窃取 


ForkJoinTask 是 Recursiveaction 的 超 类 。 它 是 从 动作 中 返回 结果 的 证 型 类 型 ， 所 以 
Recursiveaction 扩展 了 ForkJoinTask<Vvoid> 。 这 使 得 ForkJoinTask 非 常 适合 用 
MapReduce 方式 返回 数据 集中 归结 出 的 结果 。 

ForkJoinTasks 由 ForkJoinpool 调 度 安 排 , ForkJoinpoo1l 是 专 为 这 些 轻 量 任务 设计 的 新 
型 执行 者 服务 。 该 服务 维护 每 个 线程 的 任务 列表 , 并且 当 某 个 任务 完成 时 ， 它 能 把 挂 在 满 负 荷 线 
程 上 的 任务 重新 安排 到 空闲 线程 上 去 。 

采用 这 种 “工作 塞 取 ” 的 算法 是 为 了 解 次 大 小 不 同 的 任务 所 导致 的 调度 问题 。 大 小 不 同 的 任 
务 所 需 的 运行 时 间 通 常 也 会 有 很 大 差别 。 比 如 说 ， 某 个 线程 的 运行 队列 中 都 是 小 任务 ， 而 另外 一 
个 全 是 大 任务 。 如 果 小 任务 的 运行 速度 比 大 任务 快 五 倍 ， 只 处 理 小 任务 的 线程 很 可 能 在 处 理 大 任 
务 的 线程 完成 之 前 就 处 于 空闲 状态 了 。 

Java 7 实现 的 工作 等 取 机 制 精准 地 解决 了 这 个 问题 ， 并 且 在 分 云 /合并 框架 工作 的 整个 生命 周 
期 中 使 线程 池 中 的 所 有 线程 都 有 用 武之 地 , 工作 窃取 完全 是 自动 的 , 你 什么 也 不 用 做 就 能 享受 到 
它 带 来 的 好 处 。 不 需要 于 工 十 预 , 而 是 由 运行 环境 承担 更 多 工作 帮助 开发 人 员 管 理 并 发 , 这 在 Java 
7 中 已 经 不 是 什么 新 鲜 事 了 。 


册 MapReduce 是 Google 提 出 的 一 个 软件 架构 ， 用 于 大 规模 数据 集 ( 大 于 1TB ) 的 并 行 运算 。 当 前 的 软件 实现 是 指定 
一 个 Map ( 映射 ) 函数 ， 用 来 把 一 组 键 值 对 映射 成 一 组 新 的 键 值 对 ， 指 定 并 发 的 Reduce ( 化 简 ) 函数 ， 用 来 保证 
所 有 映射 的 键 值 对 中 的 每 一 个 元 素 都 共享 相同 的 键 组 。 一 一 详 者 注 
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轻易 地 简化 成 多 线程 MergeSort。 
这 里 是 一 些 可 以 用 分 支 /合并 方法 解决 的 问题 ， 
口 模拟 大 量 简单 对 象 的 运动 ， 比 如 粒子 效果 ; 
口 日 志文 件 分 析 ; 
口 从 输入 中 计数 的 数据 操作 ， 比 如 mapreduce 操 作 。 
从 另 一 个 角度 来 说 ， 图 4-10 中 这 个 被 分 解 的 问题 正 是 分 支 /合并 框架 可 以 解决 的 。 


图 4-10 分支 与 合并 


用 下 面 这 个 列表 检查 问题 及 其 于 任务 是 一 个 切实 有 效 的 方法 ， 它 可 以 确定 是 否 能 用 分 支 / 合 
并 来 解决 这 个 问题 
口 问题 的 子 任务 是 否 无 需 与 其 他 子 任务 有 显 式 的 协作 或 同步 也 可 以 工作 ? 
口 子 任务 是 不 是 不 会 对 数据 进行 修改 ， 只 是 经 过 计算 得 出 些 结果 ( 它们 是 不 是 函数 程序 员 
称 为 “纯粹 的 ”函数 的 图 数 ) ? 
口 对 于 子 任 务 来 说 ， 分 而 治之 是 不 是 很 自然 的 事 ?” 子 任务 是 不 是 会 创建 更 多 的 子 任务 ， 而 
且 它 们 要 比 派 生出 它们 的 任务 粒度 更 细 ? 
对 于 前 面 这 些 提问 ， 如 果 答 案 是 肯定 的 ， 或 者 “大 体 如 此 ， 但 有 临界 情况 ”， 那 你 的 问题 很 
可 能 适合 用 分 云 /合并 的 方式 解决 。 反 过 来 ， 如 果 答 案 是 “可 能 吧 ” 或 者 “ 算 不 上 ”"， 你 就 会 发 现 
分 支 /合并 帮 不 上 什么 忙 ， 可 能 用 其 他 的 同步 方式 更 合适 。 


注意 前 面 的 检查 列表 是 测试 某 个 问题 ( 比如 在 Hadoop 和 NoSQL 数 据 库 中 常见 的 那 种 ) 能 否 很 
好 地 用 分 支 /合并 方式 解决 的 有 效 方法 。 


想 设 计 出 优秀 的 多 线程 算法 并 不 容易 ， 分 支 /合并 方法 也 不 能 面面俱到 。 在 适用 的 领域 ， 它 
的 用 处 很 广 。 其实 归 根 结 底 ， 你 必须 要 确定 你 的 问题 是 否 适 合 这 个 框架 ， 如 果 不 适 合 ， 你 只 能 在 
性 能 时 越 的 java.util.concurrent 工 具 箱 上 构建 目 己 的 解决 方案 。 

在 下 一 节 中 ， 我 们 会 详细 讨论 经 常 被 误解 的 Java 内 存 模 型 ( Java Memory Model，JMM ),。 很 
多 Java 程 序 员 都 知道 JMM, 并 旦 在 没有 经 过 正式 介绍 的 情况 下 按 自己 的 理解 写 代 码 。 如 果 你 觉得 
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这 是 在 说 你 ， 那 么 接 下 来 的 内 容 会 帮助 你 重新 认识 JMM， 并 且 帮 你 打下 扎实 的 基础 。JMM 这 个 
话题 相当 有 深度 ， 所 以 如 果 你 急 着 进入 下 一 章 ， 可 以 先 跳 过 它 。 


4.6 Java 内 存 模 型 


Java 语 言 规范 ( JLS ) 在 第 17.4 节 中 介绍 了 JMM。 其 中 的 描述 非常 正式 ， 用 同步 动作 和 被 称 
为 偏 序 " 的 数学 结构 描述 JMM。 这 对 于 语言 理论 学 家 和 Java 规 范 的 实现 者 ( 编译 器 和 虚拟 机 的 制 
造 者 ) 来 说 非常 棒 ， 但 对 于 需要 理解 多 线程 代码 如 何 执行 的 应 用 开发 人 员 来 说 , 这 种 描述 会 让 他 
们 头 昏 脑 胀 。 

我 们 在 这 里 不 重复 规范 里 的 正式 描述 , 而 是 用 两 个 基本 概念 列 出 最 重要 的 规则 , 这 两 个 概念 
是 代码 块 之 间 的 之 前 发 生 ( Happens-Before ) 和 同步 约束 ( Synchronizes-With ) 关系 。 

口 之 前 发 生 一 一 这 种 关系 表明 一 段 代 码 块 在 其 他 代码 开始 之 前 就 已 经 全 部 完成 了 。 

口 同步 约束 一 一 这 意味 着 动作 继续 执行 之 前 必须 把 它 的 对 象 视图 与 主 内 存 进行 同步 。 

如 果 你 认真 研究 过 OO 编程 ， 应 该 听 到 过 面向 对 象 构件 的 Has-A 和 Is-A 这 两 种 表述 方式 。 一 些 
开发 人 员 发 现 ， 用 之 前 发 生 和 同步 约束 来 描述 基本 的 概念 构件 对 理解 Java 并 发 很 有 帮助 。 这 和 
Has-A 与 Is-A 的 道理 一 样 ， 但 这 两 组 概念 在 技术 上 没有 直接 关联 。 

图 4-11 中 是 一 个 易 失 性 写 人 与 后 续 的 读 取 访问 《用 于 println) 之 间 同 步 约 东 的 例子 。 


volatile long 1} 


图 4-11 同步 约束 的 例子 


JMM 的 主要 规则 如 下 。 

口 在 监测 对 象 上 的 解锁 操作 与 后 续 的 锁 操 作 之 间 存 在 同步 约束 关系 。 

口 对 易 失 性 (volatile ) 变量 的 写 人 与 后 续 对 该 变量 的 读 取 之 间 存 在 同步 约束 关系 。 
口 如 果 动 作 A 党 到 动作 B 的 同步 约束 ， 则 A 在 B 之 前 发 生 。 

口 如 果 在 程序 中 的 线程 内 A 出 现在 B 之 前 ， 则 A 在 B 之 前 发 生 。 


中 设 A 是 一 个 非 宝 集 ，P 是 A 上 的 一 个 关系 ， 若 关系 P 是 自 反 的 (对 任意 的 ae A，(a,a)eP)、 反 对称 的 (车 (a,bjeP 
生 (b，a)eP， 则 a=b ) 和 传递 的 (着 (a,bjeP，(b.cjeP， 则 (ac)eP)， 则 称 P 是 集合 A 上 的 偏 序 关系 。 比 如 实 
数 集 上 的 小 于 等 于 关系 (ag<=a; a<=b，b<=a， 则 a=b: a<=b，b<=c， 则 a<=e )， 一 -一 译 者 注 
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前 两 个 规则 通俗 来 说 就 是 “ 先 放 后 取 ”。 换 句 话 说， 一 个 线程 在 写 人 时 持 有 的 锁 要 在 其 他 操 
作 (包括 读 取 ) 能 够 获取 锁 之 前 被 释放 掉 。 

这 里 还 有 些 规 则 ， 实 际 上 是 关于 敏感 行为 的 。 

口 构造 方法 要 在 那个 对 象 的 终结 器 开始 运行 之 前 完成 (一 个 对 象 被 终结 之 前 必须 已 经 构造 

完整 )。 

口 开始 一 个 线程 的 动作 受到 这 个 新 线程 的 第 一 个 动作 的 同步 制约 。 

D Thread. join() 受 到 被 合并 的 线程 的 最 后 一 个 ( 和 其 他 全 部 ) 动作 的 同步 制约 。 

口 如 果 X 在 Y 之 前 发 生 ， 并 且 Y 在 Z 之 前 发 生 ， 则 X 在 Z 之 前 发 生 (传递 性 )。 

这 些 简单 的 规则 定义 了 内 存 和 同步 如 何 工 作 的 全 平台 视图 。 图 4-12 展 示 了 传递 性 规则 。 


而 
更 
上 
和 Y 
局 
是 

| 十 

之 前 发 生 ， 

re ee Fe pt ng ee 二 避 
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图 4-12 之 前 发 生 的 传递 性 


注意 “实际 上 ， 这 些 规则 是 JMM 做 出 的 最 低 保证 。 真 正 的 JVM 实 际 上 可 能 表现 得 更 好 。 对 于 开 
发 人 员 来 说 ， 这 可 能 是 个 陷阱 ， 因 为 某 个 特定 JVM 中 的 行为 实际 上 是 个 隐藏 在 底层 并 发 
中 的 诡异 bug， 却 很 容易 给 人 造成 错觉 ， 以 为 是 它 提供 的 安全 特性 。 


从 这 些 最 低 保证 中 ， 很 容易 可 以 看 出 不 可 变性 成 为 Java 并 发 编程 中 的 一 个 重要 概念 的 原因 。 
如 果 对 象 不 可 改变 ， 确 保 改 变 对 所 有 线程 可 见 的 相关 问题 就 不 会 出 现 。 


4.7 小结 


并 发 是 Java 平 台 最 重要 的 特性 之 一 ,扎实 的 并 发 编程 知识 对 于 一 个 优秀 的 开发 人 员 来 说 日 益 
重要 。 我 们 回顾 了 Java 并 发 的 基础 和 多 线程 系统 的 设计 原则 ， 并 讨论 了 Java 内 存 模型 和 Java 平 台 
如 何 实现 并 发 的 底层 细节 。 

更 重要 的 是 我 们 解释 java.util .concurrent 中 的 那些 类 和 接口 ， 现 代 Java 开 发 人 员 在 
编写 新 的 多 线程 代码 时 ， 更 喜欢 用 到 这 些 工 具 。 我 们 还 向 你 详细 介绍 了 Java 7 中 一 些 新 的 类 ， 如 
DinkedTransferoueue 和 分 支 /合并 框架 。 
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希望 我 们 已 经 为 你 打 好 了 基础 ， 使 你 能 够 用 java.util.concurrent 中 的 类 编写 代码 。 这 
是 本 章 内 容 的 重 中 之 重 。 尽 管 我 们 也 探讨 了 一 些 核心 理论 ,但 最 重要 的 还 是 实际 样 例 。 哪 怕 你 刚 
开始 使 用 concurrentHashMap 和 Atomic 类 ， 也 能 马上 见识 到 这 些 经 过 严格 测试 的 类 所 带 来 的 
好 处 。 

时 间 到 了 ,我们 马上 就 要 进入 下 一 主题 一 一 一 个 能 让 你 从 Java 开 发 者 中 脱 容 而 出 的 重要 主 
题 。 在 下 一 章 ， 你 会 在 Java 平 台 的 男 一 个 基础 领域 (类 加 载 和 字 节 码 ) 打下 坚实 的 基础 。 这 一 领 
域 是 很 多 讨论 平台 安全 和 性 能 特性 内 容 的 核心 , 并 巩固 了 生态 系统 内 的 很 多 先进 技术 。 所 以 对 于 
想 鹤 立 鸡 群 的 开发 人 员 来 说 ， 这 是 个 绝 佳 的 研究 课题 。 


只 自 在 读书 @3 
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类 文件 与 字 方 码 


本 章 内 容 

口 类 加 载 

口 方法 句柄 

口 解 痢 类 文件 

口 VM 字 节 码 以 及 它 的 重要 性 
口 新 的 操作 人 码 ijnvokedynamic 


要 成 为 优秀 的 Java 开 发 人 人员， 需要 深入 理解 Java 平 台 的 工作 方式 。 其 中 就 包括 类 加 载 和 JVM 
字 节 码 这 样 的 核心 特性 。 

假设 有 一 个 大 量 使 用 依赖 注入 DI ) 技术 的 应 用 程序 ， 比 如 Spring， 它 在 启动 时 出 了 问题 ， 
报 了 一 个 神秘 的 错误 信息 。 如 果 不 是 简单 的 配置 错误 问题 , 你 就 需要 了 解 如 何 实现 DI 框架 才能 跟 
味 问 题 来 源 。 也 就 是 说 你 得 明日 类 加 载 机 制 。 

或 者 假定 跟 你 合作 的 升 发 前 跑 路 了， 只 给 你 留 下 了 一 堆 编 详 过 的 代码 ， 没有 源码 ， 文 档 也 乱 
七 八 粳 的 。 你 怎么 能 知道 编译 过 的 代码 包含 了 什么 呢 ? 

最 常见 的 程序 启动 失败 错误 就 是 clas sNotFoundException 或 NoCclassDefFoundError,， 
但 很 多 开发 人 员 都 不 知道 它们 是 什么 ， 有 什么 区 别 以 及 为 什么 会 出 现 。 

本 章 重 点 就 是 这 些 与 开发 相关 的 平台 特性 。 此 外 还 会 讨论 一 些 高 级 特性 一 一 它们 是 为 Java 的 
粉丝 准备 的 ， 如 果 你 时 间 有 限 ， 可 以 跳 过 那 部 分 内 容 。 

我 们 会 从 类 加 载 的 概览 开始 ， 这 是 VM 为 运行 中 的 程序 定位 和 激活 新 类 型 的 过 程 。 其 核心 是 
在 VM 中 表示 类 型 的 Class 对 象 。 接 下 来 我 们 会 介绍 一 下 新 的 方法 句柄 API， 并 和 Java 6 中 已 有 的 
技术 ( 比如 反射 ) 进行 比较 。 

之 后 我 们 会 讨论 检查 和 分 析 类 文件 的 工具 。 用 Oracle JDK 提 供 的 javap 作 为 参考 工具 。 上 过 
类 文件 的 解剖 课 后 , 我们 会 转 而 讨论 宇 节 码 , 其 中 涉及 JVM 操 作 码 的 主要 体系 以 及 运行 时 的 底层 
操作 。 

在 你 用 字 节 码 的 知识 把 自己 武装 起 来 之 后 ， 我 们 会 深入 探讨 invokedynamic 操 作 码 ， 它 是 
Java 7 新 引 人 进 来 的 ， 为 的 是 让 非 Java 语 言 能 充分 利用 JVM 的 平台 特性 。 

我 们 先 从 类 加 载 开 始 吧 ， 这 是 一 个 将 新 的 类 合并 到 正在 运行 着 的 JVM 进 程 中 的 过 程 。 
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5.1 类 加 载 和 类 对 象 


-个 ,class 文件 定义 了 JVM 中 的 一 个 类 型 ， 包 括 域 、 方 法 、 继 承 信 息 、 注 解 和 其 他 元 数据 。 
规范 中 对 类 文件 的 格式 有 详细 描述 ， 任 何 想 在 JVM 上 运行 的 语言 都 必须 遵守 。 

类 是 平台 能 加 载 的 最 小 程序 代码 单元 。 要 将 新 的 类 加 入 到 JVM 的 当前 运行 态 中 , 有 几 步 操作 
必须 执行 。 首 先 ， 类 文件 必须 被 加 载 进来 并 连接 ,而且 必须 进行 大 量 的 验证 。 之 后 会 提供 一 个 代 
表 该 类 型 的 新 Class 对 象 给 正在 运行 的 系统 ， 并 可 以 创建 新 的 实例 。 

本 节 会 讨论 所 有 这 些 步 又 ， 并 介绍 一 下 类 加 载 岩 ,， 也 就 是 控制 整个 过 程 的 那些 类 。 我 们 先 来 
看 看 加 载 和 连接 。 


5.1.1 ”加载 和 连接 概览 


JVM 的 目的 是 使 用 类 文件 并 执行 其 中 的 字 节 码 。 要 实现 这 个 目的 , JVM 必 须 以 字 节 数据 流 的 
方式 取出 类 文件 中 的 内 容 , 并 将 其 转换 成 可 用 的 格式 加 入 运行 态 中 。 这 个 分 两 步 走 的 过 程 被 称 为 
加 载 和 连接 ， 但 连接 又 会 被 分 解 为 几 个 子 阶段 。 


: 成 2 ] 字 节 数据 流 并 给 类 的 表现 形式 解冻 , 该 过 程 一 开始 是 创建 
一 个 学 节 数 组 ， 其 内 容 通常 是 从 文件 系统 中 读 取 的 ， 然后 产生 与 所 加 载 的 类 对 应 的 Class 对 象 。 

在 这 个 过 程 中 会 对 类 做 一 些 基 本 检查 ， 但 在 加 载 过 程 结 束 时 ，Class 对 象 还 不 成 熟 , 所 以 类 也 不 
可 用 。 

连接 

加 载 完成 之 后 ， 类 必须 被 连接 起 来 。 这 一 步 妊 分 为 三 个 于 阶段 一 一 验 让 ， 淮 备 和 解析 。 验 证 
阶段 证 实 类 文件 符合 现 期 , 不 会 引起 系统 的 运行 时 错误 或 其 他 问题 。 之 后 是 类 的 准备 阶段 ， 在 类 
文件 中 引用 的 其 他 类 型 全 部 都 要 定位 到 ， 以 确保 该 类 已 准备 就 绪 。 

连接 步骤 中 各 子 阶段 之 间 的 相互 关系 如 图 $-1 所 示 。 


从 磁盘 中 读 
取 类 文件 


“ 鲜 活 ”的 类 型 
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图 5-1 ”加 载 与 连接 ( 及 连接 的 子 阶段 ) 
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5.1.2 验证 


验证 是 一 个 非常 复杂 的 过 程 ， 它 分 为 几 个 步骤 。 

首先 是 完整 性 检查 。 这 实际 上 是 加 载 过 程 中 的 一 部 分 ， 会 确保 类 文件 格式 良好 ， 可 以 连接 。 

接着 是 检查 常量 池 ( 详情 参见 5.3.3 节 ) 中 的 符号 信息 是 自 相 一 致 的 ,并 要 遵守 常量 的 基本 行 
为 准则 。 其 他 不 涉及 代码 的 静态 检查 也 要 在 这 一 阶段 完成 ， 比 如 检查 final 方 法 有 没有 被 重 写 。 

之 后 是 验证 中 最 复杂 的 部 分 一 方法 的 字 节 码 检查 。 要 检查 字 节 码 行为 良好 , 并 且 不 会 试图 
摆脱 VM 的 环境 控制 。 下 面 是 一 些 主要 检查 。 

是 否 所 有 方法 都 遵守 访问 控制 关键 字 的 限定 。 

D 方法 调用 的 参数 个 数 和 静态 类 型 是 否 正 确 。 

确保 字 节 码 不 会 试图 滥用 堆栈 。 

确保 变量 使 用 之 前 被 正确 初始 化 了 ， 

检查 变量 是 否 仅 被 赋予 恰当 类 型 的 值 。 

做 这 些 检查 是 出 于 性 能 方面 的 考虑 , 这 样 可 以 加 快 解释 码 的 运行 速度 , 运行 时 就 不 用 再 做 这 
些 检查 了 。 同 时 还 简化 了 运行 时 把 字 节 码 编译 为 机 器 码 的 过 程 (即时 编译 ， 详 情 参见 6.6 节 )。 

准备 

类 的 准备 包括 分 配 内 存 和 准备 好 初始 化 类 中 的 静态 变量 , 但 不 会 现在 初始 化 变量 , 也 不 会 执 
行 任何 VM 字 节 码 。 

解析 

解析 会 促使 VM 检 查 类 文件 中 所 引用 的 类 型 是 不 是 都 是 已 知 的 类 型 。 如 果 有 运行 时 未 知 的 类 
型 ， 那 它们 也 需要 被 加 载 进来 。 这 些 可 见 的 未 知 类 型 会 再 次 引发 类 加 载 过 程 。 

一 旦 需要 加 载 的 其 他 类 型 全 被 定位 并 解析 完成 ，VM 就 可 以 初始 化 那个 最 初 要 加 载 的 类 。 这 
时 所 有 静态 变量 都 可 以 被 初始 化 , 所 有 静态 初始 化 代码 块 都 会 运行 。 现 在 你 运行 的 字 节 码 就 是 来 
自 新 加 载 进来 的 类 里 的 。 这 一 步 完成 之 后 ， 类 的 加 载 就 已 全 部 完成 ， 类 也 就 可 以 使 用 。 


5.1.3 Class 对象 


连接 和 加 载 过 程 的 最 终结 果 是 一 个 Class 对象 , 用 于 表示 加 载 并 连接 起 来 的 新 类 型 。 尽 管 出 
于 性 能 方面 的 考虑 ，cClass 对 象 只 是 在 要 求 的 地 方 做 了 初始 化 ， 但 现在 它 在 VM 中 完全 生效 了 。 
代码 可 以 继续 执行 了 , 它 可 以 使 用 新 类 型 并 创建 新 实例 。 此 外 ，class 对 象 提 供 了 一 些 不 错 的 方 
法 ， 比 如 getSsuperclass()， 可 以 用 它 返 回 class 对 象 的 父 类 。 

Class 对 象 可 以 和 反射 API 一 起 实现 对 方法 、 域 、 构 造 方 法 等 类 成 员 的 间接 访问 。class 对 
象 中 有 对 类 成 员 Method 和 Fiela 对 象 的 引用 。 反 射 API 可 以 用 这 些 对 象 实 现 对 它们 的 间接 访问 。 
图 5-2 是 这 种 结构 的 高 层 视 图 。 

运行 时 的 哪个 部 分 会 定位 并 连接 宇 节 流 以 生成 新 的 加 载 类 ? 在 下 一 个 主题 中 , 我们 会 讨论 这 
个 问题 ， 即 能 够 完成 这 些 工作 的 类 加 载 右 ， 它 是 由 抽象 类 classLoadezr 的 子 类 们 组 成 的 。 
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图 5-2 ”class 对 象 与 Method 引 用 


5.1.4 类 加 载 器 


Java 平 台 里 有 几 个 经 典 的 类 加 载 器 ， 它 们 在 平台 的 启动 和 常规 操作 过 程 中 承担 不 同 的 
任务 ， , 
D 根 (或 引导 ) 类 加 载 器 一 通常 在 VM 启动 后 不 久 实例 化 ， 一 般 用 本 地 代码 实现 最 好 把 ”四 
它 看 做 VM 的 一 部 分 。 它 的 作用 通常 是 负责 加 载 系统 的 基础 JAR ( 主要 是 rt .jar ),， 而且 
它 不 做 验证 工作 。 
口 扩展 类 加 载 器 一 一 用 来 加 载 安 装 时 自 带 的 标准 扩展 。 一 般 包括 安全 性 扩展 。 
口 应 用 (或 系统 ) 类 加 载 器 一 一 这 是 应 用 最 广汉 的 类 加 载 器 。 它 负责 加 载 应 用 类 。 在 大 多 
数 SE (Java 标准 版 ) 的 环境 中 ， 主 要 工作 都 是 由 它 来 完成 。 
口 定制 类 加 载 器 一 一 在 更 复杂 的 环境 中 ， 比 如 EE (Java 企业 版 ) 或 比较 复杂 的 SE 框架 ， 通 
常会 有 些 附加 ( 即 定制 ) 的 类 加 载 器 。 有 些 团 队 基 至 为 他 们 的 革 个 应 用 程序 编写 了 特定 
的 类 加 载 器 。 
除了 核心 任务 ， 类 加 载 器 还 经 常 要 从 JAR 文 件 或 classpath 中 加 载 资源 ( 不 是 类 文件 ， 比 如 图 
片 或 配置 文件 )。 


Ne Ee El NR ts ee te a 
1 ”全 了 一 工具 类 加 载 器 Se 呈 
在 EMMA 测 试 赣 盖 工具 ( http://emma sourceforge nety ) 中 使 用 的 一 个 类 加 载 器 可 以 作为 加 
载 时 转化 的 例子 。 
当 为 了 加 上 额外 的 测试 辅助 信息 而 加 载 类 时 ，EMMA 的 类 加 载 器 会 修改 字 节 码 。 当 在 这 


些 代 码 上 运行 测试 用 例 时 ，EMMA 会 记录 测试 用 例 实 际 测试 了 哪些 方法 和 代码 分 支 。 从 这 些 
记录 中 ,开发 人 员 能 看 出 对 一 个 类 的 单元 测试 是 否 全 面 。 关 于 测试 和 禾 盖 ， 在 11 和 12 章 还 有 更 
多 的 相关 讨论 。 


有 些 框架 和 代码 还 经 表 会 使 用 带 有 额外 属性 的 专用 ( 甚至 用 户 自 定义 的 ) 类 加 载 器 。 这 些 类 
加 载 器 经 常会 在 加 载 时 对 字 节 码 进行 转换 ， 我 们 在 第 1 章 有 提 到 过 。 
图 $-3 中 是 类 加 载 器 的 继承 层级 以 及 不 同 加 载 器 之 间 的 相互 关系 。 
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图 $-3 ”类 加 载 器 层级 
我 们 先 来 看 一 个 专用 类 加 载 器 的 例子 一 一 如 何 用 类 加 载 实现 依赖 注入 。 
5.1.5 示例: 依赖 注入 中 的 类 加 载 器 


DI 有 两 个 核心 思想 。 

口 系统 内 的 功能 单元 要 靠 依赖 项 和 配置 信息 才能 正常 发 挥 作用 。 

口 在 对 象 自身 的 上 下 文 里 ， 很 难 表 示 依 赖 项 ， 即 使 可 以 ， 也 很 复杂 难 懂 。 

你 脑海 中 应 该 浮现 出 一 幅 包 含 了 行为 ， 配置 和 依赖 项 信息 ( 它们 处 在 对 象 之 外 ) 的 画面 。 这 
于 分 通常 锌 称 为 对 象 的 运行 时 路 线 。 

第 3 章 以 Guice 框 架 为 例 介 绍 了 DI。 本 节 中 我 们 会 讨论 利用 类 加 载 器 实现 DI 的 方式 , 但 这 种 方 
式 与 Guice 不 同 ， 实 际 上 它 更 像 简 化 版 的 Spring 框架 。 

在 这 个 想象 出 来 的 DI 框架 下 ， 我 们 像 这 样 启 动 应 用 程序 ; 

java -cp <CLASSPATH> org.framework .DIMain /path/to/config.xml 

CLASSPATH 中 必须 包含 DI 框架 的 JAR 文 件 以 及 在 config.xml 文 件 中 引用 的 所 有 类 ( 以 及 它们 
的 所 有 依 环 项 ) 的 JAR 文 件 。 

我 们 改写 了 前 面 一 个 类 似 的 例子 一 一 代码 清单 3-7 中 的 服务 ， 结 果 如 清单 5-1 所 示 。 


sp class HOL LOG or viced 


rivate AgentFinder finder = null) 
| 空 的 构造 方法 
public HollywoodServiceDI() {|} | 


public void setFinder (AgentFinder 上 inaer | 
this.finder = finder:; 
| setter 方 法 
public List<cAgent> getFriendlyAMgents() I | 同 代 码 清 单 3-7 
} 
public List<Agent> filterAgents (lList<Agent> agents, String agentType) | 


of | 同 代码 清单 3-7 


从 晶 在 读书 @ 


WWwwW .Zizidiary .com 


5.2 使 用 方法 自 柄 11] 


为 了 将 它 置 于 DI 框架 的 管理 之 下 ， 还 需要 一 个 配置 文件 : 


<beanass 
<bean id="agentFinder" class="Wgqjd.ch03.WebServiceAgentFinder" 


oi 


<bean id="hwSservice" class="Wgjd.ch0o5.HollywoodServiceDI" 
p:finder-ref="agentFinder"/> 
</beans> 


在 这 种 方式 中 ，DI 框 架 利 用 配置 文件 来 确定 要 构造 的 对 象 。 这 个 例子 需要 hwservice 和 
agentFinder 两 个 bean ， 框 架 会 为 每 个 bean 调 用 空 构 造 方 法 ,之 后 是 setter 方 法 ( 比如 为 
HollywoodServiceDI 的 依赖 项 AgentFinder 调 用 setFinder() )。 

这 说 明 类 加 载 分 为 两 个 阶段 。 第 一 阶段 由 应 用 类 加 载 器 完成 ， 负 责 加 载 DIMain 及 其 引用 的 
类 。 然 后 DIMain 开 始 运行 ， 并 在 main() 的 参数 中 得 到 配置 文件 的 位 置 。 

这 时 候 ， 框 架 已 经 在 JVM 中 运行 起 来 了 ， 但 config.xml 中 指定 的 用 户 类 还 磁 都 没 碰 呢 。 实 际 
上 ， 在 DIMain 检 查 配 置 文件 之 前 ， 框 架 不 可 能 知道 要 加 载 什 么 类 。 

要 局 动 config.xml 中 指定 的 应 用 配置 ， 需 要 类 加 载 的 第 二 阶段 。 这 要 用 到 定制 的 类 加 载 磊 。 
首先 ， 要 检查 config.xml 文 件 的 一 臻 性， 确保 它 没 有 错误 。 然 后， 如 果 吾 无 差错 ， 定 制 的 类 加 载 
器 就 会 试图 从 cLASsPATH 中 加 载 指定 类 型 。 如 果 任 何 一 步 失 败 了 ， 和 整个 过 程 就 会 被 中 止 。 

如 果 成 功 了 ，DI 框 架 可 以 继续 创建 所 需 的 实例 ， 并 调用 实例 上 的 setter 方 法 。 如 果 这 些 都 顺 
利 完成 了 ,程序 上 下 文 就 开始 运行 了 。 

我 们 简单 介绍 了 一 下 Spring 风格 的 DI 方 式 ， 其 中 大 量 使 用 了 类 加 载 。 在 Java 拉 术 中 ,还 有 很 
多 要 用 到 类 加 载 郑 及 其 相关 技术 的 领域 。 下 面 是 一 些 众 所 周知 的 例 于 : 

口 插件 染 构 ; 

口 厂商 提供 的 或 目 主 研发 的 框架 ; 

口 从 非 正 常 位 置 ( 非 文件 系统 或 URL ) 获取 类 文件 ; 

DD Java EE:; 

口 任何 需要 在 JVM 进 程 已 经 启动 后 加 入 新 的 、 未 知 代码 的 情况 下 。 

我 们 对 类 加 载 的 讨论 就 到 此 为 止 。 让 我 们 进入 下 一 节 ， 探 讨 Java 7 为 满足 反射 等 需求 而 提供 
的 新 API。 


5.2 使 用 方法 句柄 


如 果 你 不 熟悉 Java 的 反射 API ( class、Method、Field 和 它们 的 朋友 )， 可 以 大 致 浏览 一 下 
(其 至 跳 过 ) 这 一 节 的 内 容 。 可 如 果 你 的 代码 库 中 有 很 多 反射 代码 ， 那 么 你 一 定 要 认真 读 一 读 ， 
因为 它 介 绍 了 Java 7 中 取得 相同 效果 的 新 办 法 ,而且 所 用 的 代码 更 简洁 。 

Java 7 为 间接 调用 引 方法 人 了 新 的 AP[。 其 中 的 关键 是 java.lang.invoke 包 ， 即 方法 句柄 。 
你 可 以 把 它 看 做 反射 的 现代 化 方式 ， 但 它 不 像 反 射 API 屠 样 有 时 会 显得 元 长 、 繁 重 和 粗糙 。 
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反射 中 有 很 多 套路 化 的 代码 。 如果 你 写 过 一 些 反 射 代 码 , 就 不 会 忘记 必须 一 次 又 一 次 地 用 
Class[] 指 向 内 省 方法 的 参数 类 型 ， 并 把 该 方法 的 参数 都 封装 成 Object[] ， 还 要 捕捉 各 种 讨 
质 的 异常 以 防 出 错 ， 而 且 反 射 代码 看 起 来 也 很 不 直观 .。 

通过 将 反射 代码 转移 到 方法 句柄 ， 可 以 去 掉 套 路 化 的 代码 ， 提 高 代码 的 可 读 性 , 这 是 大 势 
所 趋 。 


方法 句柄 是 将 invokedynamic ( 详情 参见 5.5 节 ) 引 人 JVM 项 目 中 的 一 部 分 。 但 其 作用 不 仅 
限于 invokedynamic 的 应 用 案例 ， 在 框架 和 和 常规 用 户 代码 中 也 有 用 武之 地 。 接 下 来 我 们 会 先 介 
绍 方法 句 林 的 基本 技术 ;之 后 会 给 出 一 个 例子 与 现 有 的 各 种 方式 进行 比较 ,并 总 结 出 其 中 的 差异 。 


D2.1 MethodHandle 


什么 是 MethodHandle? 它 是 对 可 直接 执行 的 方法 (或 域 、 构 造 方 法 等 ) 的 类 型 化 引用 ,这 
是 标准 答案 。 还 有 一 种 说 法 : 方法 句柄 是 一 个 有 能 力 安全 调用 方法 的 对 象 。 

下 面 我 们 要 获取 一 个 带 有 两 个 参数 的 方法 ( 但 我 们 可 能 连 这 个 方法 的 名 字 都 不 知道 ) 的 方法 
句柄 ， 之 后 调用 对 象 cbj 上 的 句柄 ， 传 人 参数 arg0 和 argl: 

MethodHandle mh = getTwoArgMH () : 

MyType Tet 

try 1 

ret = mh.invokeExact (obij, arg0, argl):; 
} catch (Throwable el 1 


e.printSstackTrace(); 


) 

这 种 能 力 有 些 像 反 射 ， 还 有 些 像 4.4 节 介绍 的 callable 接 口 。 实 际 上 ，callable 是 对 方法 
调用 能 力 建 模 的 早期 尝试 。 但 它 只 适用 于 不 带 参 数 的 方法 。 为 了 满足 现实 情况 中 不 同 参 数组 合 和 
调用 的 可 能 ， 我 们 需要 编写 带 有 特定 参数 组 合 的 其 他 接口 。 

Java 6 中 有 很 多 这 种 代码 ， 接 口 四 处 草 延 ， 让 开发 人 员 万 分 音 恼 ( 比如 耗 光 保存 类 信息 的 
PermGen 内 存 一 一 见 第 6 章 )。 相 比较 而 育 ， 方 法 句柄 则 适用 于 任何 方法 签名 ， 不 需要 产生 那么 多 
小 类 。 这 要 归功 于 新 引信 的 MethodType 类 。 


5.2.2 MethodType 


MethodType 是 表示 方法 签名 类 型 的 不 可 变 对 象 。 每 个 方法 句柄 都 有 一 个 MethodType 实 例 , 用 
来 指明 方法 的 返回 类 型 和 参数 类 型 。 但 它 没有 方法 的 名 字 和 “接收 者 类 型 "， 即 调用 的 实例 方法 的 
用 MethodType 类 中 的 工厂 方法 可 以 得 到 MethodType 实 例 。 这 里 有 几 个 例子 : 


MethodType mtToSstring = MethodType .methodType (String.class); 
MethodType mtSetter = MethodType.methodType (void.clagss, Obiject .class); 
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MethodType mtStringComparator = Method 
String.class, String.class):; 


这 些 MethodType 实例 分 别 表 示 tostring() ，setter 方 法 (object 类 的 成 员 ) 和 
Comparator<String> 定 义 的 compareTo () 方 法 的 类 型 签名 。MethodType 实 例 一 般 都 遵循 相 
同 的 模式 ， 第 一 个 传人 的 参数 是 方法 的 返回 类 型 ， 随 后 的 参数 是 方法 参数 的 类 型 ( 跟 Class 对 象 
一 样 )， 如 下 所 示 : 

MethodType .methodType (RetType .class, ArgOType .class, ArglType .class, ...); 

你 看 ,现在 可 以 用 普通 对 象 来 表示 不 同 的 方法 签名 了 , 不 需要 再 逐一 为 它们 定义 新 类 型 。 这 
也 在 最 大 程度 上 保证 了 类 型 安全 性 ,而 且 办 法 还 很 简单 。 如 果 你 想 知 道 某 个 方法 句柄 能 否 用 特定 
的 参数 集 调 用 ， 可 以 检查 该 句柄 的 MethodType。 

现在 你 应 该 明日 MethoadType 是 如 何 解 决 接口 沁 滥 的 问题 
类 中 方法 的 方法 句柄 吧 。 


5.2.3 查找 方法 句柄 


下 面 的 代码 展示 了 如 何 得 到 指 阿 当前 类 中 tostring() 方 法 的 方法 句柄 。 注 意 ,mtToeString 
和 tostring() 的 签名 完全 一 致 ， 返 回 类 型 为 string， 没有 参数 。 也 就 是 说 相应 的 MethodType 
实例 是 MethodaType .methodTypel(String.class)。 


代码 清单 5-2 ”查找 方法 句柄 
public MethodHandle getToStringMa() | 
MethodHandle mh ， 


MethodType mt = MethodType.methodType (String.class}: | 获取 上 下 文 
MethodHandles.Lookup lk = MethodHandles, 1ookupl(); 


vpe .methodType (int .class, 


了 , 接 下 来 就 去 看 看 怎么 得 到 指 问 


ee = lk.findvirtual (getClass()}, "tostring", mt};} 
SS 
return mh; 
| 
取得 新 的 方法 句 顶 要 用 lookup 对 象 ， 比 如 代码 清单 5-2 中 的 1kx。 这 个 对 象 可 以 提供 其 所 在 环 
境 中 任何 可 见方 法 的 方法 句柄 。 
要 从 lookup 对 象 中 得 到 方法 句柄 ， 你 需要 给 出 持 有 所 需 方 法 的 类 、 方 法 的 名 称 ， 以 及 跟 你 所 
需 的 方法 签名 相 匹 配 的 MethodType。 


注意 在 坦 找 上 下 文 (lookup context ) 中 可 以 得 到 任何 类 型 ( 包括 系统 类 型 ) 中 的 方法 自 栖 。 
当然 ， 如 果 要 从 没有 关联 的 类 中 取得 句柄 ， 查 找 上 下 文中 只 能 看 到 或 取得 public 方 法 的 
和 句柄。 就 是 说 方法 句柄 总 是 在 安全 管理 之 下 安全 使 用 没有 反射 中 setaccessiblel) 
那 种 破解 方法 。 
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现在 你 已 经 拿 到 了 方法 句柄 ， 接 下 来 自然 是 执行 它 。 方 法 句柄 API 为 此 提供 了 两 个 方法 : 
invokeExact() 和 invoke()。invokeExact() 方 法 要 求 其 参数 类 型 与 底层 方法 所 期 望 的 参数 
类 型 完全 匹配 。invoke () 方 法 会 在 参数 类 型 不 太 正 确 时 做 些 修改 ， 以 使 其 与 底层 方法 参数 相 匹 
配 ( 比如 在 需要 时 进行 装 逢 或 拆 箱 )。 

接 下 来 我 们 会 给 出 一 个 长 一 点 儿 的 例子 , 说 明 如 何 使 用 方法 句柄 取代 过 去 的 技术 ,比如 反射 
和 小 型 代理 类 。 


5.2.4 示例 ; 反射 、 代 理 与 方法 名 棉 


如 果 你 曾经 处 理 过 满 是 反射 的 代码 库 ， 就 会 深 知 反射 代码 所 带 来 的 痛 语 了 。 在 本 市 中 ,我 们 
要 问 你 证 明 方 法 句柄 可 以 取代 很 多 套路 化 的 反射 代码 ， 会 让 你 的 编码 生 淮 更 轻松 。 

代码 清单 5-3 是 改编 自前 面 章节 的 例子 。ThreadPoolManager 人 负责 将 新 任务 分 配给 线程 池 ， 
和 代码 清单 4-15 稍 有 不 同 。 它 还 能 取消 正在 运行 的 任务 ， 但 是 个 私有 方法 。 

为 了 闻 明 方法 句 档 和 其 他 技术 之 间 的 差别 ， 我 们 给 出 了 从 外 部 访问 类 的 私有 方法 cancel |) 
的 三 种 办 法 . makeReflective、makeProxy 和 makeMh。 我 们 还 展示 了 两 种 Java 6 技术 : 反射 和 
代理 类 。 并 且 和 基于 MethodHandle 的 方式 进行 了 比较 。 我 们 用 到 了 一 个 读 取 队列 的 任务 
oueueReaderTask 实现 了 Runnable 接 口 ) .你 可 以 在 本 音源 码 中 找到 oueueReaderTask 实 现 。 


代码 清单 5-3 ”三 种 访问 方式 
public class ThreadPoolManager | 


private final ScheduledFxecutorService stpe = 
Executors.newScheduledThreadPool (2); 
private final BlockingQueue<WorkUnit<String>»> lbaq,; 


public ThreadPoolManager (BlockingQueue<WorkUnit<Strings> lbq ) | 
lbdo = lbq ; 


public ScheduledFuture<?> runlQueueReaderTask msgReader}) | 
msgReader .SetQUueue (lbq).:; 
return stpe.scheduleAtFixedRate (megReader, 10, 10, 
Timetnit .MILLISECONDS) ; 


private void cancel (final ScheduledFuturec?» hndl) | 
stpe.schedule (new Runnable() | ] 要 访问 的 私有 方法 
public void run() {| hndl .cancel [true}); | 
}, 10, TimeUnit .MILLISECONDS); 
} 
public Method makeReflective() | 
Method meth = null; 
try I 
Class<?>[] argTypes = new Class[] { ScheduledFuture.class }; 
meth = ThreadPoolManager ,class.getDeclaredMaethodl"cancel", 
argTypes); 
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meth.sethAccessible (true).,; 


} catch (IllegalArgumentException | NoSuchMethodException 要 求 访问 私有 
| SecurityException e) { 方法 
e.printStackTrace ();} 
| 
return meth; 


| 


public static class Canmcelpreoxy I 
private CancelPproxy() | } 
public void invoke (ThreadPoolManager mae , ScheduledFuture<?> hndl ) | 
mae .cancel {hndal ); 
} 
} 
public CancelProxy makeProxy() | 
return new CancelProxy\(}:; 
| 
public MethodHandle makeMh() | 建 
MethodHandle mh; 创 
MethodType desc = MethodType .methodType (void.class, 
scheduledFuture.class); 
try { 查找 MethodHandle 


mh = MethodHandles .lookup!) 
.findVvirtual (ThreadPoolManager .class, "cancel", desc}); 


} catch (NoSuchMethodException | IllegalAccessException el 1 
Ehrow (ASSErti1ionBrror)new AssgertionBrror'() .initcause (e); 


} 


return mh; 


| 
| 


这 个 类 提供 了 三 个 访问 私有 方法 cancel () 的 方法 。 实 际 上 ， 一 般 实现 时 只 会 用 一 个 ， 我 们 
是 为 了 讨论 它们 之 间 的 差别 才 全 都 列 了 出 来 。 
下 面 是 使 用 这 些 方法 的 例子 。 


代码 清单 54 ”使 用 这 些 访问 方法 


private void cancelUsingReflectionlScheduledFuture<?> hndl) | 
Method meth = manager.makeReflectivel), 


try { 
System.out .println("With Reflection"):; 
meth.invoke (hndl):; 
} cateh (IllegalAccessException | IllegalArgument Exception 
| InvocationTargetException el) | 
e.printstackTrace (); 
| 
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private void cancelUsingProxy [ScheduledFuture<?> hndl) 1 
CancelProxy proxy = manager ,makeProxy |),; 


通过 代理 调用 
是 静态 类 型 的 


system.out .println("With Proxy"); 
proxy .invoke (manager, hndl); 


private void cancelUsingMH lScheduledFuture<?> hndl) { 
MethodHandle mh = manager ,makeMh (1) | 


| 
ery 方法 签名 必须 
System.out .println("With Method Handle"), 完全 一 致 
mh.invokeExact (manager, hndl):; 


} catch {Throwable el | 


e.printstackTrace(); 必须 捕捉 
Throwable 


| 


BlockingQueue<WorkUnit<String>> lbq = new LinkedBlockingQueue<>(); 
manager = new ThreadPoolManager (lbq), 4 


final QueueReaderTask msgReader = new QueueReaderTask (100) | 
Override 
public void doAction(lString mag ) | 


if (msg l= null System.out .println("Msg recvd: "+ msg |); 
hndl = manager .run (m8gReader)}:; . 取消 任务 


这 几 个 cancelUsing 方 法 都 有 一 个 ScheduledFuture 参 数 ， 所 以 你 可 以 用 前 面 的 代码 试验 
不 同 的 取消 方法 。 实 际 上 ， 作 为 API 的 使 用 者 ， 你 可 以 不 用 去 管 这 是 如 何 实现 的 。 
在 下 一 节 中 ， 我 们 会 告诉 你 API 或 框架 开发 人 员 用 方法 句柄 取代 其 他 方式 的 原因 。 


Db.2.5 为 什么 选择 Meth 和 


在 上 一 节 中 我 们 看 了 一 个 把 方法 句柄 用 在 Java 6 中 使 用 反射 和 代理 的 地 方 的 例子 。 这 引出 了 
一 个 问题 : 为 什么 要 用 方法 句柄 取代 过 去 的 处 理 方 式 ? 

从 表 S5-1 可 以 看 出 ， 反 射 最 大 的 优势 就 是 人 们 珊 悉 它 。 代 理 对 于 人 徐 单 用 例 可 能 更 容易 理解 ， 
但 我 们 认为 方法 句柄 在 这 两 方面 做 得 都 是 最 棒 的 。 我 们 强烈 推荐 你 使 用 方法 句柄 。 


表 5-1 Java 的 方法 间接 访问 技术 比较 


特 性 反 射 代理 方法 句柄 
访问 控制 必须 使 用 setaccesible() 。 内 部 类 可 以 访问 受 限 方法 ”在 恰当 的 上 下 文中 对 所 有 方 
会 被 安全 管理 器 禁止 法 都 有 完整 的 访问 权限 。 和 安 

全 管理 器 没有 冲突 


类 型 纪律 (Type discipline) 役 有 。 不 匹配 就 抛 出 异常 静态 的 。 过 于 严格 。 为 了 在 运行 时 是 娄 型 安全 的 ,。 不 占 
存储 全 部 的 代理 上 娄 ， 可 能 ”用 PermGen 


需要 很 多 PermGen 
性 能 跟 其 他 的 比 算 慢 的 眼 其 他 方法 较 用 一 样 快 。 “力求 跟 其 他 方法 调用 一 样 快 
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方法 句柄 还 有 一 个 特性 , 可 以 从 静态 上 下 文中 确定 当前 类 。 如 果 你 曾经 编写 过 这 样 的 日 志 代 
码 ( 比如 log4j ): 

Logger lgr = LoggerFactory .可 所 蕊 LO 可 吕 已 工 (MyClass.class); 

你 应 该 知道 这 样 的 代码 很 脆弱 。 如 果 它 被 重 构 进 超 类 或 子 类 中 , 显 式 声明 的 类 名 就 会 有 问题 。 
然而 在 Java 7 中 ， 你 可 以 这 样 写 ; 

Logger lgr = LoggerFactory.getLogger (MethodHandles.loo0kupl) .lookupClass'()):; 

在 这 行 代码 中 , 可 以 把 lookupclass () 看 成 用 在 静态 上 下 文中 的 getclass(), 这 在 处 理 日 
志 框 架 之 类 的 场合 中 特别 有 用 ， 因 为 通常 每 个 用 例 都 有 自己 的 logger。 

带 着 新 掌握 的 方法 句柄 技术 ， 我 们 去 检查 一 下 类 文件 的 底层 细节 和 使 其 变 得 有 意义 的 工具 。 


5.3 ”检查 类 文件 


类 文件 是 二 进 制 块 , 所 以 想 直接 和 它 打交道 不 太 容 易 。 但 有 很 多 时 候 你 会 发 现 必须 和 类 文件 
交手 。 
比如 说 , 为 了 在 运行 时 更 好 地 监控 ( 比如 通过 JMX ) 应 用 程序 , 你 需要 加 上 额外 的 公共 方法 。 
重新 编译 和 再 次 部 署 看 起 来 顺利 完成 了 ， 但 检查 管理 API 时 却 发 现 设 有 那些 方法 。 又 进行 了 几 次 
构建 和 部 署 还 是 没有 发 现 。 
为 了 找 出 部 署 问题 ， 你 需要 检查 一 下 javac 产 生 的 类 文件 是 不 是 你 想 要 的 那个 。 还 有 时 候 你 
需要 研究 那些 没有 源码 的 类 文件 ， 以 验证 文档 中 是 不 是 真有 你 所 怀疑 的 错误 。 
对 于 类 似 的 任务 ， 你 必须 用 工具 检查 类 文件 的 内 容 。 好 在 标准 的 OracleJVM 中 有 javap 这 个 
工具 ,用 它 来 探视 类 文件 内 部 和 反 汇 编 类 文件 非常 得 心 应 手 。 
我 们 一 开始 会 先 介 绍 javap， 以 及 为 检查 类 文件 而 设置 的 各 种 基本 参数 。 接 下 来 会 讨论 方法 
名 称 和 类 型 在 JVM 内 部 的 一 些 表 示 方 式 。 然 后 看 一 下 篆 量 池 ， 它 是 JVM 的 “ 藏 宝箱 ”"， 对 于 理解 
字 节 码 如 何 工 作 非 常 重要 。 
5.3.1 介绍 javap 
javap 的 用 处 很 多 ， 既 能 看 类 声明 了 什么 方法 ， 又 能 输出 字 节 码 。 我 们 来 看 一 下 javap 最 简 
单 的 用 途 ， 在 第 4 章 讨论 的 微 博 Update 上 坛 一 下 。 
$$ javap wgjd/ch04/Update.clasa 
Compiled freoem "Update. java" 
public class wgjd.ch04.Update extends java.lang.0bject | 
public wgjd.cho04a .Author getAuthor'(); 
public java.lang.Sstring getUpdateText (); 
public int hashcode|().; 
public boolean equals (java.lang.Obiject):; 
public java. lang.sString toString(}); 
wgqijd.cho4.Update (wgid.ch04 .Update$Builder, wgijd.ch04 .Update); 
} 
默认 情况 下 ，javap 会 显示 访问 权限 为 public、protected 和 默认 ( 即 包 级 protectea ) 
级 别 的 方法 。 加 上 -p 选 项 后 还 可 以 显示 private 方 法 和 域 。 
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5.3.2 方法 签名 的 内 部 形式 


JVM 内 部 用 的 方法 签名 和 javap 显 示 出 来 供 人 阅读 的 形式 不 太一 样 。 随 着 我 们 对 JVM 的 不 断 
深入 ， 这 些 内 部 名 称 出 现 将 更 加 频繁 。 如 果 你 赶 时 间 ， 可 以 跳 过 这 一 节 。 但 请 记 住 它 ， 因 为 你 可 
能 还 要 回来 参考 这 些 内 容 。 

在 紧凑 形式 中 ， 类 型 名 称 是 经 过 压缩 的 。 比 如 int 是 用 r 表 示 的 。 这 些 紧 凑 形式 有 时 被 称 为 
类 型 描述 符 。 表 5-2 中 是 类 型 描述 符 的 完整 列表 。 

表 5-2 ”类 型 描述 符 


描 述 符 类 型 
B byte 
Cc char (16 位 Unicode 字 符 ) 
DD double 
F float 
I int 
可 Long 
Le< 业 型 名 称 > 引用 类 型 (比如 Ljava/lang/String; 用 于 字符 唐 ) 
S short 
2 boolean 
[ array-of 


某 些 情况 下 ， 类 型 描述 行 可 能 比 类 型 名 称 还 要 长 ( 比如 Ljava/lang/Object 就 比 object 
长 )， 但 类 型 描述 符 是 完全 限定 的 ， 所 以 可 以 直接 解析 。 

javap 还 有 一 个 有 用 的 选项 -s， 可 以 输出 签名 的 类 型 描述 待 , 所 以 你 没 必 要 用 那个 表 自 己 做 
转换 。 你 可 以 使 用 javap 高 级 一 些 的 方法 来 显示 我 们 之 前 看 过 的 一 些 方 法 的 签名 : 


$ Javap -Ss wgijd/ch04/Update .class 
Compiled from "Update.Java”" 
public class wgijd.ch04.Update extends java.lang.OQbject 1 
public wgjd.ch04.Author getAuthor(), 
Siagnature: ()Lwgjd/ch04/Author; 


public java.lang.String getUpdateText () ) 
Siagnature: ()Lijava/lang/String; 


public int compareTo(lwgjd.ch04.Update),; 
Signature: (Lwgjd/ch04/Update;)I 


BuBlic int hashCodel}., 
Signature: (}I 


如 你 所 见 ， 方 法 签名 中 的 所 有 类 型 都 是 用 类 型 描述 符 表 示 的 。 
在 下 一 节 中 你 会 看 到 类 型 描述 符 的 另 一 个 用 途 。 它 会 出 现在 类 文件 中 非常 重要 的 部 分 一 一 常 
项 闻 。 
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5.3.3 常量 池 


谓 量 池 是 为 类 文件 中 的 其 他 ( 常量 ) 元 素 提 供 快捷 访问 方式 的 区 域 。 如 果 你 研究 过 C 或 Perl 
之 类 的 语言 , 应 该 知道 符号 表 , 对 于 JVM 来 说 , 常量 池 就 类 似 于 符号 表 。 但 和 其 他 语言 不 同 , Java 
没有 完全 开放 对 常量 池 中 信息 的 访问 。 

为 了 不 纠缠 于 过 多 的 细节 ,我 们 用 一 个 非常 简单 的 例子 来 演示 线程 池 。 下 面 是 一 个 简单 的 “ 游 
戏 围 栏 ” 或 者 叫 “ 演 算 本 ”类 。 我 们 在 这 个 类 的 run() 里 面 写 一 点 代码 ， 就 可 以 快速 测试 Java 的 
请 法 特性 或 类 库 。 


代码 清单 5-5 ”游戏 围栏 样 例 类 


package wgijd.ch04,; 


public class ScratehIimpl |{ 
private static ScratchIimpl inst = null; 


private ScratcechIimpl(}) I 
| 


private void run() | 


| 

public static void main(sString[] arge) 1 
inst = new ScratchImpl ():; 
inast .runt():; 


| 
} 
要 查看 常量 池 中 的 信息 ， 可 以 用 javap-v。 这 个 命令 还 会 输出 很 多 其 他 信息 ,不 过 我 们 只 关 
注 常量 池 中 的 条 目 。 
#1 = Class #2 1/ wgjd/ch04/ScratchIimpl 
#2 = Utf8 wgjd/ch04/ScratchImpl 
#3 = Clasa | /java/langyobject 
#4 = UEtB java/lang/Qbject 
#5 = Utf8 inst 
#6 = Ut£8 Lwgjd/ch04/SscratchIimpl:; 
#7 三 Utf8 Clinits 
#8 = Utf£8 (YY 
#9 = Utf£8 Code 
#10 = Fieldref 样 工 . 拓 工 工 
= // wgijd/ch04/SscratchIimpl .inst:Lwgjd/ch04/sScratchIimpl,; 
#11 = NameAndType #*5:#6€ /:/ instance:Lwgjd/cho0a/scratchImpl,; 
#12 = Utf8 LineNumberTable 
#13 = Utf8 LocalVariableTable 
#14 = Utf8 <inits> 
#15 = Methodref 大 3 . 非 16 // java/lang/Qbject."<init>":(})V 
#16 = NameAndType 伴 工 生 ; 林 昌 jo "einit>": ()V 
#17 = Utf8a this 
样 】 日 = Utf8 run 
#19 = UEEB ([Lijava/lang/String;}V 
#20 = Methodref 村 二 . 非 卫 工 /:/ wgqgjd/ch04/ScratchImpl .run: (}V 
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#21 = NamehndType #18 :#8 i run:(}v 
#22 = Utfg args 
机 23 = 【七 上 上 日 [Liava/lang/string; 
赃 24 = Utfa maln 
要 25 = Methodref 六 了. 粒 1 扎 :i waijd/cho4/ScratchImpl, "<inits": ()V 
#26 = Methodref 间 1 . 埋 227 
:i wgijd/cho0a4/ScratchIimpl .run: ([Ljava/lang/String;}Vv 
#27 = NameAndType #18:#19 /:/ run: (lLjava,/lang/string;)}V 
#28 三 Utfa SOUurceFile 
#29 = Utf8 scratchImpl .java 


= /j/ wagild/cho4/ScratchIimpl .run: ([Liava/lang/string;)})Vv 
#27 = NameAndType #18:#19 CC// run:l[Liava/lang/string;}Vv 


#28 二 Utf£8 SOUrceFile 
#29 三 Utf8 ScratchIimpl .java 


如 你 所 见 ， 和 常量 池 中 的 条 目 是 带 有 类 型 的 。 它 们 还 会 相互 引用 ， 比 如 说 ， 一 个 类 型 为 class 
的 条 目 会 引用 类 型 为 Ut £8 的 条 目 。 而 Utf8 的 条 目 是 个 字符 串 ， 所 以 class 条 目 引 用 的 Utf8 条 目 
应 该 是 类 的 名 称 。 

表 5-3 是 可 能 出 现在 常量 池 中 的 条 目 集 。 在 讨论 常量 池 中 的 条 目 时 ， 有 时 会 用 coNsTANT 前 


编 ， 比 如 CONSTANT Class。 


表 5-3 常量 池 条 目 


名 称 描 述 
Class 类 常量 。 引 用 类 的 名 称 (Ut£f8 条 目 ) 
Fieldref 定 人 对 域 ，5| 用 该 域 的 Class 和 NameAndType 
Methodref 定 浆 方法。 引用 读 方 法 的 class 和 NameAnadType 
InterfaceMethodref 定义 接口 方法 。3 引 用 读 方 法 的 class 和 NameandType 
string 字符 串 常 量 。 引 用 保存 字符 的 Ut £8 常量 
Integer 整 型 常量 (4 字 节 ) 
Float 浮 点 常量 (4 字 布 ) 
Long 长 整 型 常量 (8 字 节 ) 
Double 双 精 度 浮 点 型 常量 (8 字 节 ) 
NameAndType 拉 述 和 名称 和 类 型 对 。 类 型 引用 一 个 保存 类 型 描述 符 的 Ut£8 条 目 
Utf£B8 一 个 表示 LUtf8 编 码 的 字符 的 二 进 制 字 节 流 
InvokeDynamic (Iava 7 中 新 引入 的 ) 见 5.5 节 
MethodHandle (Java 7 中 新 引信 的 ) 描述 MethodHandle 常 量 
MethodType (Java 7 中 新 引 八 的 ) 描述 MethodType 常 量 


你 可 以 用 这 个 表格 从 演算 类 的 常量 池 中 看 到 常量 解析 的 例子 。 比 如 条 目 #10 中 的 Fieldref。 

要 解析 一 个 域 ， 你 需要 名 称 、 类 型 ， 还 有 它 所 在 的 类 ， 和 机 0 的 值 是 #1 .#11， 这 就 是 说 常量 #11 
来 目 类 #1]1。 在 输出 中 可 以 很 容易 看 出 #1 确实 是 一 个 class 类 型 的 常量 ， 并 且 #11 是 NamennaType。 
#1 指 问 ScratchImpl 类 本 身 ， #11 是 #S:#6 一 一 个 名 称 为 inst 的 ScratchImp1 变 量 。 所 以 综合 来 
看 ,所 0 指向 scratchImp1l 类 内 部 的 自身 静态 变量 inst( 你 可 能 已 经 从 清单 5-6 的 输出 中 猪 出 来 了 )。 
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在 类 加 载 过程 中 的 验证 环节 , 有 一 步 是 检查 类 文件 中 的 静态 信息 是 否 一 致 的 。 前面 的 例子 是 
运行 时 在 加 载 新 类 时 要 做 的 完整 性 检查 。 
对 于 类 文件 的 基本 结构 ,我 们 已 经 讨论 的 差不多 了 , 接 下 来 要 进入 下 一 话题 一 一 字 节 人 码 。 理 
解 源 码 如 何 变 成 字 节 码 会 对 你 理解 代码 如 何 运 行 有 很 大 的 帮助 ,在 学 习 第 6 章 以 及 后 面 的 章节 时 ， 
还 能 引号 你 更 加 深入 地 了 解 平台 的 能 力 。 


5.4 字 节 码 


到 目前 为 止 ,在 我 们 的 讨论 中 , 字 节 码 一 直 有 点 幕后 工作 者 的 意思 。 我 们 先 来 回顾 一 下 对 它 
已 经 有 了 哪些 了 解 ， 然 后 再 对 它 进行 详细 介绍 : 

口 字 节 码 是 程序 的 中 间 表 示 形 式 ， 介 于 人 类 可 读 的 源码 和 机 器 码 之 间 。 

口 字 节 码 是 由 源码 文件 中 的 javac 产 生 的 。 

口 某 些 高 层 语 言 特性 在 编译 时 已 经 从 字 节 码 中 去 掉 了 。 比如 说 Java 的 循环 结构 ( for .while 

等 ) 在 字 节 码 中 就 被 转换 成 了 分 支 指令 。 

口 每 个 操作 码 都 由 一 个 字 节 表示 因此 被 叫做 宇 节 码 )。 

口 字 节 码 是 一 种 抽象 表示 法 ， 不 是 “ 某 种 虚拟 CPU 的 机 器 码 ”。 

口 字 节 码 可 以 进一步 编译 成 机 器 码 ， 通 党 是 “即时 编译 ”。 

字 节 码 解 释 起 来 有 点 像 先 有 鸡 还 是 先 有 和 弄 的 问题 。 要 彻底 搞 清 楚 状 况 ， 你 既 要 懂 字 节 码 ， 又 
要 明白 执行 它 的 运行 时 环境 。 

这 是 一 个 循环 依赖 ,为 了 解决 这 个 问题 ,我 们 先 来 探索 一 个 相对 简单 的 例子 。 即 使 这 一 次 你 
不 太 明 白 ， 也 可 以 在 后 续 章 节 读 到 更 多 字 节 人 码 相 关内 容 时 再 回来 看 看 。 

在 例子 之 后 , 我 们 会 给 出 一 些 与 运行 时 环境 相关 的 上 下 文 和 JVM 操 作 码 的 目录 ( 其 中 包括 用 
于 数学 计算 、 调 用 、 人 快捷 形式 之 类 的 字 节 码 )。 最 后 ， 我 们 会 用 另外 一 个 基于 字符 串 拼接 的 例子 
来 结束 。 现 在 就 先 去 看 看 如 何 检 查 .class 文 件 的 字 节 码 吧 。 


5.4.1 示例 ; 反 编译 类 


用 带 有 -c 选 项 的 javap 可 以 对 类 进行 反 编译 。 我 们 会 以 代码 清单 5-5 中 的 演算 类 为 例 ， 主 要 
检查 方法 之 内 的 字 节 码 。 我 们 还 会 加 上 -p 选 项 ， 以 便 能 见 到 私有 方法 内 的 字 节 码 。 

我 们 一 节 一 节 的 来 一 javap 输 出 的 每 一 部 分 都 有 很 多 信息 ， 很 容易 让 人 不 堪 重负 。 首 先 ， 
让 我 们 先 看 头 部 。 这 里 没什么 特别 出 人 意料 或 让 人 喜出望外 的 ; 


$ javap ~C - wgjd/ch04/ScratchImpl .class 

compiled from "Scratchlmpl .java" 

public class wgjd.ch04.ScratchImpl extenda java,lang.Qbject | 
private static wogjd.cho4.ScratchImpl inst; 


接 下 来 是 静态 块 。 变 量 的 初始 化 就 放 在 这 里 ， 所 以 这 表示 inst 被 初始 化 为 nul1 了 。 看 起 来 
putstatic 可 能 是 一 个 把 值 放 到 静态 域 中 的 字 节 码 。 
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static {}; 


Code: 
0: aconst null 
l1: putstatic #10 1 Field inst:Lwogjd/ch0a/ScratchIimpl:; 


二 return 
代码 前 面 的 数字 表示 从 方法 开始 算 起 的 字 节 码 偏 移 量 。 所 以 宇 节 1 是 putstatic 操 作 码 ， 字 
节 2 和 3 表示 一 个 16 位 的 常量 池 索 引 ， 这 个 16 位 索引 在 这 里 的 值 是 10， 表 示 该 值 ( 此 处 为 null ) 
会 存在 常量 池 的 条 目 检 0 所 指明 的 域 中 。 从 字 节 码 流 开始 的 第 4 个 字 节 是 return 操 作 符 ， 表 明 这 
个 代码 块 结束 了 。 
接 下 来 是 构造 方法 。 
private wgijd.ch0o4.SsScratcehIimpl '():; 
Code: 
0: aload 0 
1:; invokespecial #15 /:/ Method java/lang/Object."<inits":(}V 
4; return 


在 Java 中 ,void 构造 方法 总 会 隐 式 调用 超 类 中 的 构造 方法 。 这 从 上 面 的 字 节 码 里 就 能 看 出 来 
invokespecial 指 令 。 一 般 来 说 ， 任 何方 法 调用 都 会 转换 成 VM 的 某 一 调用 指令 。 
在 run() 方 法 中 没有 代码 ， 因 为 这 只 是 一 个 空白 的 演算 类 。 
private Vvoid runt()}; 
Code: 
0: return 


在 main 方 法 中 ， 你 初始 化 了 inst， 还 做 了 点 对 象 创建 。 这 说 明了 辨识 通用 字 节 码 的 基本 
模式 : 


public static void mainljava.lang.string([]}}; 
code: 
0: new | /:/ class wgjd/ch04/SscratchImpl 
3: dup 


4: invokespecial #21 :i: Method "ainity": (})V 
这 种 3 个 字 节 人 友 指 令 的 模式 一 new、dup 和 一 个 <init> 的 jnvokespecial 都 表示 创建 
新 实例 。 
操作 码 new 只 为 新 实例 分 配 内 存 。dup 复 制 栈 硕 上 的 元 泰 。 要 完整 创建 该 对 象 ， 你 需要 调用 
构造 方法 的 代码 块 。<init> 方 法 中 包含 构造 方法 的 代码 ， 所 以 可 以 用 invokespecial 调 用 那 段 
代码 。 我 们 继续 看 main 方 法 中 其 余 的 字 节 码 : 


了 putstatic 林 工 剖 :| Field inst:Lwagjd/ch04/ScratchImpl:; 
10: getatatic #10 :i Field inst:Lwgjd/choa/scratchIimpl:; 


13; invokespecial #22 :i: Metheod run: (}V 
1]: return 
} 
指令 7 保存 刚刚 创建 的 单 例 实例 。 指 令 10 把 它 放 回 到 栈 硕 上 ， 这 样 指令 13 就 可 以 调用 它 上 面 
的 方法 了 。 广 意 ， 因 为 调用 的 run() 蚌 私有 方法 ， 所 以 13 是 invokespecial。 私 有 方法 不 能 重 
写 ， 所 以 不 能 用 Java 的 标准 虚拟 查询 。 大 多 数 方法 调用 都 会 转换 成 invokevirtual 指 令 。 


1 二 
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注意 通常 来 说 ，javac 产 生 的 字 节 码 没 有 经 过 特别 优化 ， 是 非常 简单 的 表示 形式 。 基 本 策略 
是 由 JIT 编 译 回 来 完成 天 部 分 的 优化 工作 , 所 以 简单 直 自 的 起 点 对 它们 是 很 有 帮助 的 .VM 


实现 者 表示 ,“ 字 节 码 就 应 该 傻 傻 的 "”， 这 是 他 们 对 从 源 语言 产生 的 字 节 码 的 总 体感 觉 。 


接 下 来 我 们 要 讨论 字 节 码 所 需 的 运行 时 环境 , 之 后 会 介绍 用 来 描述 字 节 码 指令 主要 “家 庭 成 
员 ” 的 表格 ， 其 中 包括 加 载 /存储 ， 数 学 计算 ， 执行 控制 ， 方 法 调用 和 平台 操作 。 然 后 我 们 会 讨 
论 操 作 码 可 能 的 快捷 形式 ， 最 后 会 再 给 出 一 个 例子 。 

5.4.2 ”运行 时 环境 
因为 JVM 使 用 堆栈 机 ， 所 以 理解 堆栈 机 的 操作 对 理解 字 节 码 至 关 重要 。 
李 市 码 


图 5-4 ”将 栈 用 于 数学 运算 
JVM 与 硬件 CPU ( 比如 x64 或 ARM 心 片 ) 最 显著 的 差别 在 于 它 设 有 处 理 器 寄存 器 ， 而 是 用 栈 
完成 所 有 的 计算 和 操作 。 有 时 候 也 这 也 被 称 为 操作 数 栈 ( 或 计算 堆栈 )- 图 $-4 展 示 了 如 何 用 操作 
数 栈 完成 两 个 int 数 值 的 相 加 运算 。 
正如 我 们 前 面 讨论 过 的 ， 当 一 个 类 被 链接 进 运行 时 环境 时 ， 它 的 宇 节 码 会 受到 检查 ， 并且 其 
中 很 多 验证 都 可 以 归结 为 对 栈 中 类 型 模式 的 分 析 。 


注意 ，” 栈 中 的 值 只 有 类 型 正确 时 对 它 的 处 理 才能 生效 。 比 如 ， 如 果 我 们 把 对 一 个 对 象 的 引用 压 
入 栈 ， 然 后 试图 将 其 作为 int 型 进行 数学 计算 ， 就 可 能 会 发 生 未 定义 或 粳 粒 的 事情 。 类 加 
载 过 程 中 的 验证 阶段 会 进行 广泛 的 检查 ， 以 确保 新 加 载 的 类 中 不 会 有 滥用 栈 的 方法 。 这 
样 做 能 够 防止 系统 接受 了 损坏 (或 恶意 ) 的 类 并 引发 问题 。 


方法 在 运行 时 需要 一 块 内 存 区 域 作为 计算 堆栈 来 计算 新 值 。 另外, 每 个 运行 的 线程 都 需要 一 
个 调用 堆栈 ( 栈 跟踪 中 会 报告 的 那个 栈 ) 来 记录 当前 正在 执行 的 方法 。 在 某 些 情况 下 ,这 两 个 栈 
会 有 交互 。 看 下 面 这 行 代码: 


return 3 + petRecords.getNumberoOfPetsl("Ben"),; 


只 自 在 读书 @3 
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要 计算 出 这 行 代码 的 结果 ， 需 要 把 3 压 人 操作 数 栈 。 然 后 调用 方法 计算 Ben 有 多 少 只 宠物 。 为 此 ， 
你 需要 把 接收 对 象 (方法 属 主 ， 即 petRecords ) 压 人 计算 堆栈 ， 要 传人 的 所 有 参数 尾随 其 后 。 

然后 invoke 操 作 符 会 调用 方法 getNumberOfPets () , 把 控制 权 移 交 给 被 调用 的 方法 , 刚刚 
进入 的 方法 会 出 现在 调用 堆栈 中 。 但 进入 新 方法 后 ， 需 要 启用 不 同 的 操作 数 栈 , 所 以 已 经 在 调用 
者 的 操作 数 栈 中 的 值 不 可 能 影响 第 调用 方法 的 计算 结果 。 

在 getNumberOftPets1() 气 成 时 ， 返 回 绪 果 会 被 放 到 调用 者 的 操作 数 栈 中 ， 进 程 中 与 
getNumberOfPets() 相 关 的 部 分 也 会 从 调用 堆栈 中 移 走 。 然 后 相 加 运算 可 以 得 到 两 个 值 并 把 它 
们 加 在 一 起 。 

现在 我 们 开始 审视 宇 广 码 。 这 是 个 大 课题 , 而且 有 很 多 特殊 情况 ,所 以 我 们 即将 呈现 的 只 是 
主要 特性 的 概览 ， 而 不 是 完整 的 介绍 。 


5.4.3 ”操作 码 介 绍 


JVM 字 节 人 码 由 操作 码 (opcode ) 序列 构成 ， 每 个 指令 后 面 可 能 会 跟着 一 些 参 数 。 操 作 码 希望 
看 到 栈 处 于 指定 状态 中 ， 然 后 它 对 栈 进 行 转 换 ， 把 参数 移 走 ， 放 人 结果 。 

每 个 操作 码 都 由 一 个 单 宇 节 值 表示 ， 所 有 最 多 只 能 有 255 个 操作 码 。 当 前 仅 用 了 200 个 左右 。 
对 我 们 来 说 ， 把 它们 全 列 出 来 有 点 儿 太 多 了 ,好 在 大 多 数 操作 码 都 可 以 归 为 几 大 族 系 。 我 们 会 逐 
一 对 这 些 族 系 进行 讨论 ， 帮助 你 理解 它们 。 还 有 一 些 操作 码 不 好 界定 应 该 归 为 哪 一 族 系 , 但 好 在 
你 不 会 经 常 遇见 它们 。 


注意 JVM 不 是 纯粹 的 面向 对 象 运行 时 环境 一 它 支 持原 始 类 型 。 这 在 某 些 操作 码 族 系 中 有 所 
体现 其 中 一 些 基本 操作 码 类 型 ( 比如 存储 和 相 加 ) 要 有 一 些 变 体 ， 在 处 理 原始 类 型 
时 会 有 所 不 同 。 


操作 码 表 有 四 列 : 

口 名 称 : 这 是 操作 码 类 型 的 通用 名 称 。 大 多 数 情况 下 ， 都 会 有 几 个 相关 的 操作 码 在 做 类 似 
的 事情 。 

口 参数 :操作 码 的 和 参数。 以 i 打头 的 参数 是 用 来 作为 常量 池 或 局 部 变量 中 的 查询 索引 的 几 个 
字 节 。 如 果 有 更 多 的 此 类 参数 ， 它 们 会 合并 在 一 起 ， 所 以 i1，i2 表 示 “ 从 这 两 个 字 节 中 生 
成 一 个 16 位 的 索引 "。 如 果 参 数 出 现在 括号 里 ， 就 表明 不 是 所 有 形式 的 操作 码 都 会 使 用 它 。 

口 推 栈 布局 : 它 展示 了 栈 在 操作 人 码 执行 前 后 的 状态 。 括 号 中 的 元 素 表 明 不 是 所 有 形式 的 操 
作 人 码 都 使 用 它们 ， 或 者 这 些 元 素 是 可 选 的 ( 比如 调用 操作 码 ), 

口 描述 : 操作 人 码 的 用 处 。 

我 们 从 表 5-4 中 拿 过 来 一 行 代 码 做 例子 ,检查 一 下 操作 码 getfielad 的 条 目 。 这 个 操作 码 用 于 

从 对 象 的 域 中 该 出 一 个 值 。 
getfield il, i2 [obj] 一 [val] 从 剖 顶 问 对 单 的 常量 池 中 取出 指定 位 置 的 域 ， 
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第 一 列 给 出 了 操作 码 的 名 字 : getfield。 后 面 一 列 说 明 在 字 节 码 流 中 有 两 个 参数 跟 在 操作 
码 后 面 。 这 些 参 数 合 在 一 起 构成 一 个 16 位 的 值 ， 可 以 用 来 从 常量 池 里 找到 想 要 的 域 ( 记 住 常量 池 
的 索引 总 是 16 位 的 )。 

堆栈 布局 那 一 列表 明 在 找到 栈 顶 端 对 象 的 类 的 常量 池 中 的 索引 位 置 之 后 , 该 对 和 象 被 移 除 , 它 
的 位 置 被 那个 域 所 替代 。 

这 种 把 移 走 对 和 象 作为 操作 一 部 分 的 模式 是 一 种 让 和 字 扩 人 码 变 得 紧 痊 的 办 法 , 没有 又 开 的 清理 工 
作 ， 也 不 用 记 着 要 挪 走 处 理 完 的 对 象 实例 。 


5.4.4 加载 和 依存 操作 码 


加 载 和 储存 操作 码 这 个 族 系 负责 将 值 加 载 到 栈 或 检索 值 。 表 S$-4 给 出 了 加 载 /储存 族 系 的 主要 
操作 。 
表 5-4 加载 和 依存 操作 和 码 四 
名 称 参数 堆栈 布局 找 述 
load (41} [] 。 [val] 从 局 部 变量 加 载 值 (原始 型 或 引用 型 ) 到 栈 上 。 有 快捷 
形式 ， 井 且 有 针对 不 同业 型 的 变 体 
ldc i1 [] 。 [vall] 从 池 中 加 载 常 量 到 栈 上 ， 针 对 不 同类 型 有 不 同 的 变 体 ， 
并 且 范 围 广泛 
store (41) [val] 。 [] 把 值 【原始 型 或 引用 型 ) 从 进程 的 栈 中 移 走 ， 存 到 局 部 
变量 中 。 有 快捷 形式 ， 有 针对 不 同类 型 的 变 体 
dup [val] 。 [val, val] 复制 栈 顶 部 的 值 ， 有 不 同形 式 的 变 体 
getfield hE. [obij] * [vall] 从 权 质 部 对 象 的 常量 字 中 得 到 指定 位 置 的 域 
putfielad 11, i2 [obj,.val] 。 [] 把 值 放 人 对 象 在 党 量 字 中 指定 位 置 的 域 上 


前 面 提 过 , 加载 和 储存 指令 有 很 多 不 同形 式 的 变 体 。 比 如 用 来 把 双 精 度数 从 局 部 变量 加 载 到 
栈 上 的 aload 操 作 码 ， 以 及 用 来 把 对 象 引 用 从 栈 弹 出 到 局 部 变量 中 的 astore 操 作 码 。 


5.4.5 数学 运算 操作 码 


这 些 操作 符 在 栈 上 执行 数学 运算 。 它 们 从 栈 顶 端 取出 参数 并 进行 计算 。 这 些 参 数 ( 总 是 原始 
型 ) 必须 完全 匹配 ， 但 平台 提供 了 很 多 对 原始 型 进行 类 型 转换 的 操作 码 。 表 5-5 给 出 了 基本 的 数 
学 运算 操作 码 。 
类 型 转换 (east ) 操作 码 的 名 称 非 常 短 ， 比 如 i2a 是 把 int 转 为 double 的 操作 码 。 需 要 特别 
说 明 的 是 ， 类 型 转换 操作 码 中 并 没有 cast， 所 以 在 表 5-5 中 用 括号 把 它 括 了 起 来 。 
表 5-5 数学 运算 操作 码 
名 称 参 数 堆栈 布局 描 述 | 
add [vall, val2] 。[res] 把 栈 顶端 的 两 个 值 相 加 【必须 是 相同 的 原始 类 型 ) ， 并 把 结果 
存在 栈 中 。 有 快捷 形式 ， 有 针对 不 同类 型 的 变 体 
sub [vall，val2] 。[res] ”把 栈 顶 端的 两 个 值 相 减 (必须 是 相同 的 原始 类 型 】， 并 把 结果 
存在 栈 中 。 有 快捷 形式 ， 有 针对 不 同类 型 的 变 体 
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名 称 参 数 堆栈 布局 


( 续 ) 
描 述 


把 栈 顶 端的 两 个 值 相 除 (必须 是 相同 的 原始 类 型 ) ， 并 把 结果 


存在 栈 中 。 有 快捷 形式 ， 有 针对 不 同类 型 的 变 体 


把 栈 顶 端的 两 个 值 相 乘 (必须 是 相同 的 原始 类 型 ) ， 并 把 结果 


存在 栈 中 。 有 快捷 形式 ， 有 针对 不 同类 型 的 变 体 


div [vall, val2] » [res] 
ml [vall, val2] » [res] 
(cast) [walue] » [res] 


把 值 从 一 种 原始 类 型 转换 为 另外 一 种 。 每 一 种 可 能 的 类 型 转换 


都 有 对 应 的 形式 


5.4.6 ”执行 控制 操作 码 


如 前 所 述 ， 高 级 语言 的 控制 结构 在 JVM 字 节 码 中 没有 出 现 。 相 反 ， 流程 控制 是 由 很 少 的 几 个 


原始 指令 完成 的 ， 如 表 5-6 所 示 。 


囊 5-6 
名 称 参 数 堆栈 布局 
半生 bl, b2 [wall, wal2] =。 
[] 或 fvall] 。 [] 
goto bl, b2 [] = [ 
js "bi, b2 [] 。 [ret] 
ret 索引 [】 =» [【] 
tableswitch { 俯 情 况 而 定 } [index)] »。 [] 
lookupswitch { 依 情 i 况 而 定 } [key] *» [] 


流程 控制 操作 码 


描 。 述 
如 果 符 合 特定 条 件 ， 则 跳 转 到 特定 分 支 的 偏 移 处 


无 条 件 地 跳 转 到 分 支 仿 移 处 。 有 宽大 形式 

跳 到 本 地 子 流 程 中 ， 并 把 返回 地 址 (下 一 个 操作 
码 的 偏 移 地 址 ) 放 到 栈 中 。 有 宽大 形式 

返回 到 索引 的 局 部 变量 所 指向 的 偏 移 地 址 

用 于 实现 switch 

用 于 实现 switch 


就 像 用 于 查找 常量 的 索引 字 节 ,参数 bl1、b2 用 于 构造 方法 内 部 的 字 节 码 跳 转 地 址 。jsr 指 令 
用 于 访问 主流 程 之 外 一 个 目 成 体系 的 字 节 码 区 域 〈 偏 移 地 址 可 能 在 方法 的 主 字 节 码 之 外 ) 在 革 


些 情况 下 ， 比 如 在 异常 处 理 块 中 ， 可 能 会 用 到 它 。 


goto 和 jsr 指 令 的 宽大 形式 要 用 4 个 字 节 的 参数 , 并 且 所 构造 的 偏 移 量 大 于 64KB。 但 这 并 不 


常用 。 


5.4.7 调用 操作 码 


调用 操作 码 中 有 四 个 操作 码 可 以 处 理 普 通 的 方法 调用 ， 还 有 一 个 Java 7 中 新 出 的 特别 操作 码 
invokedynamic ( 5.5 节 有 更 多 细节 )。 这 五 个 方法 调用 操作 人 码 如 表 5-7 所 示 。 


表 5-7 ”调用 操作 三 


参数 堆栈 布局 拱 ” 述 加 
DT 人 RE il, 工 2 [lvall, .1] 。T] 调用 一 个 静 志 方法 下 
invokevirtual 141 2 [obj, (vall, ...})}] = []】 调用 一 个 “常规 ”的 实例 方法 


由 自在 读书 他 
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( 续 ) 


名 称 参 数 堆 机 布局 描 述 


invokeinterface il, i2,cCount, 0 [obj, (vall, ...))] =» [] 调用 一 -个 接口 方法 
invokespecial 和 [obhj, (vall, ...)] = |{] 周 用 一 个 “特殊 ”的 实例 方法 
invokedynamic il, i2, OQ,0 [Valls ss] = 1[) 动态 调用 ， 见 $.5 节 


在 调用 操作 码 中 ， 有 两 个 地 方 需要 注意 。 第 一 个 是 invokeinterface 中 多 出 来 的 参数 。 这 
些 参 数 基 于 历史 原因 和 问 后 莱 容 而 产生 ,但 现在 已 经 用 不 到 了 。 在 invokedynamic 的 参数 中 多 
出 来 的 两 个 0 是 基于 前 向 兼容 而 产生 的 。 

男 外 一 个 是 常规 和 特别 实例 方法 调用 之 间 的 差别 。 常规 调用 是 虚拟 的 。 这 就 是 说 被 调用 的 方 
法 是 在 运行 时 按照 标准 的 Java 方 法 重 写 规则 查找 的 。 特 殊 调用 不 考虑 重 写 。 在 两 种 情况 下 这 很 重 
要 ， 即 私有 方法 和 超 类 方法 的 调用 。 在 这 两 种 情况 下 ， 你 不 想 触 发 重 写 规则 ,所 以 需要 不 同 的 调 
用 操作 码 处 理 这 种 情况 。 


5.4.8 ”平台 操作 操作 码 


平台 操作 族 系 的 操作 码 包 括 new， 用 于 分 配 新 的 对 象 实例 ， 还 有 与 线程 相关 的 操作 码 ， 比 如 
monitorenter 和 monitorexit。 详细 内 容 请 参见 表 5-8。 
平台 操作 人 码 用 来 控制 对 象 生命 周期 ， 比 如 创建 新 对 象 并 锁 住 它 们 。 一 定 要 注意 ，new 操 作 码 
只 分 配 存储 空间 。 对 象 构建 的 高 层 概 念 还 包括 运行 构造 方法 内 的 代码 。 
表 5-8 平台 操作 码 
名 称 参 数 堆栈 布局 描 述 
new 11, i2 [] » [obj] 为 新 对 象 分 配 内 存 ， 类 型 由 指定 位 置 的 常量 确定 
monitorenter [obj] » [【] 锁 住 对 象 
monitorexit [obj] *。 [【] 解锁 对 象 
在 字 节 人 码 这 一 级 ,构造 方法 被 转换 成 市 有 特殊 名 称 <init> 的 方法 。 这 不 能 由 用 户 代 码 调用 ， 
但 可 以 由 字 节 码 调用 。 这 便 形 成 了 一 个 与 对 象 创建 直接 相关 的 不 同 字 节 码 模 式 : new 之 后 跟着 一 
个 dup， 然后 是 一 个 调用 <inits 方 法 的 ijnvokespecial。 


5.4.9 ”操作 码 的 快捷 形式 

为 了 节省 字 节 , 很 多 字 节 码 都 有 快捷 形式 。 通常 对 某 些 局 部 变量 的 访问 要 比 其 他 的 访问 更 加 
频 和 或， 所 以 用 特殊 的 操作 但 来 表示 “在 局 部 变量 上 直接 执行 常见 操作 ” 便 很 有 价值 。 因 此 加 载 / 
存储 族 系 中 出 现 了 aload_0 和 dstore_2 这 种 操作 码 。 

我 们 来 检查 一 下 其 中 的 理论 ， 再 来 看 一 个 例子 。 
5.4.10 示例: 字符 串 拼 接 

我 们 给 演算 类 中 加 点 料 , 来 阐明 几 个 稍微 高 级 点 的 字 节 码 , 下 面 的 例 了 于 会 涉及 字 节 码 主 要 族 
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系 中 的 大 多 数 。 


别 忘 了 , Java 中 的 字符 串 是 不 可 变 的 。 那 在 用 + 运算 符 把 两 个 字符 串 拼 在 一 起 时 发 生 了 什么 ? 


你 必须 创建 一 个 新 字符 串 ， 但 实际 上 可 能 不 止 这 么 简单 。 
看 一 下 修改 了 run() 方 法 之 后 的 演算 类 ， 
private void run{string[] args) 1 
String str = "foo",; 
i£f (args.length > 0) str = argel0]; 
Syatem.out .println("this is my string: " + str); 
} 
这 个 简单 方法 对 应 的 字 节 但 为 : 


$$ javap -€ -Bp waijd/ch04/scratchImpl .class 
Compiled from "ScratchIinmpl .Java" 


private void runl(ijava.lang.stringl])}):; 
Code: 
0; ldc #17 /! String foo 
2: aStore 2 


3: aload 1 
4: arraylength 


5: 1iLfle 12 # 有 A 


如 果 传 人 数组 尺寸 小 于 等 于 0， 跳 到 指令 12。 


8: aloagd 1 
9: iconast 0 
10: aaload 
1l: astore 2 
l2: getastatic #19 
= // Field ava/lang/SsSystem.out:Lijava/lioco/Printstream; 


上 上面 这 行 是 访问 System.out 的 字 节 码 。 


15: new #5 /'/ class java/lang/stringBuilder 
18: dup 
19: ldec 林 也 了 六 String this is my string: 


21: invokespecial #29 
se // Method java/lang/stringBuilder,. "<inits": (Ljava/lang/String;})Vv 
24: aload 2 
25: invokevirtual #32 
= // Method java/lang/stringBuilder .append 
= (Lijavallang/string;}Liava/lang/stringBuilder; 
28; invokevirtual #36 
“ // Method java/lang/sStringBuilder.toString: (}Ljava/lang/string:; 


这 些 指令 展示 本 拼接 字符 串 的 创建 过 程 。 特 别 是 15~23 表 示 对 象 创建 ( new 、dup 和 
invokespecial ) 的 指令 ,但 在 这 个 例子 中 dup 之 后 还 有 一 个 lac ( 加 载 常 量 )。 这 种 模式 表明 


字 节 码 调 用 的 是 一 个 非 空 构造 方法 ， 在 此 是 StringBuilder (String)。 


这 个 结果 一 开始 可 能 有 些 出 乎 你 的 意料 。 你 只 是 想 把 一 些 字符 串 拼 在 一 起 , 但 到 了 底层 突然 
变 成 了 创建 额外 的 StcringBuilder 对 象 ， 并 调用 appenda() ， 然 后 又 调用 kostring()。 这 是 因 
为 java 中 的 字符 串 是 不 可 变 的 。 你 不 能 通过 拼接 修改 字符 串 对 象 ， 所 以 必须 创建 新 的 对 象 。 


stringBuilder 是 完成 这 个 任务 的 便捷 方法 。 
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最 后 是 调用 相应 的 方法 输出 结果 : 
31: invokevirtual #40 

= // Method java/io/Printstream.println: (Lijava/lang/string;}V 
34: return 


最 终 ， 输 出 字符 捉 拼 好 了 了 ， 你 可 以 调用 println() 方 法 。 因 为 此 时 栈 顶 部 的 两 个 元 素 是 
[System.out,<output string>] ， 所 以 这 是 在 System.ocout 之 上 调用 的 。 就 跟 你 在 看 表 5$-7 
(定义 了 有 效 的 jnvokevirtual 的 堆栈 布局 ) 时 所 预期 的 一 样 。 

要 成 为 一 名 真正 优秀 的 Java 开 发 人 员 ， 你 应 该 找 几 个 自己 写 的 类 用 javap 运 行 一 下 ， 并 学 会 
识别 通用 的 字 节 人 码 模 式 。 现 在 ， 让 我 们 带 着 对 字 节 码 的 简单 了 解 ， 进 入 下 一 主题 一 一 Java 7 中 重 
要 的 新 特性 ijnvokedynamic。 


5.5 invokedynamic 


本 他 主要 针对 Java 7 中 最 复杂 的 新 特性 之 一 。 尽 管 这 个 特性 十 分 强大 ， 但 它 并 不 是 给 所 有 开 
发 人 员 准 备 的 , 它 只 会 出 现在 非常 高 级 的 用 例 中 。 目前 来 看 , 这 个 特性 是 为 框架 开发 人 员 和 非 Java 
语言 准备 的 。 

也 就 是 说 如 果 你 对 平台 底层 如 何 运转 不 感 兴趣 ,对 新 的 字 节 码 细 节 毫 不 关心 , 请 跳 到 小 结 部 
分 或 直接 进入 下 一 章 ， 没 关系 的 。 

如 果 你 还 在 , 很 好 。 接 下 来 我 们 可 以 向 你 介绍 invokedynamic 的 出 现 是 多 么 不 同 寻常 。Java 
7 引入 了 一 个 胃 新 的 字 节 码 ， 这 在 Java 世 界 中 可 是 从 来 没有 过 的 大 事件 。 这 个 字 节 码 新 秀 就 是 
invokedynamic, 一 种 新 的 调用 指令 ,是 用 来 做 方法 调用 的 。 它 可 以 用 来 告诉 VM 必须 延迟 确定 
要 调用 哪个 方法 。 也 就 是 说 VM 不 用 像 往 常 一 样 在 编译 或 达 接 时 就 敲定 所 有 细节 。 

相反 ， 需 要 什么 方法 在 运行 时 决定 。 通 过 调用 一 个 辅助 方法 来 确定 应 该 调用 哪个 方法 。 


ma J . | | | ee 


Sm 


在 Java 7 中 ，Ja 
直接 编译 成 invokedynar 
这 些 动态 能 力 。 

invokedynamic 有 是 为 非 java 语 言 准备 的 。 添 加 它 是 为 了 让 动态 语言 能 够 利用 Java 7 VM， 
不 过 有 些 聪明 的 Java 框 架 也 找到 了 让 invokedynamic 为 它们 服务 的 办 法 。 


本 和 言 还 不 he . 没有 哪 个 Java 表 达 式 会 被 5 
tic。 人 们 希望 Java 8 会 增加 更 多 的 语言 结构 ( 比如 默认 方法 ) 来 使 用 


我 们 在 本 市 中 会 给 出 invokedynamic 的 工作 细节 ， 还 会 给 出 一 个 详细 的 例子 一 一 反 编 译 一 
个 利用 新 字 扩 人 码 的 调用 点 。 注 意 ， 要 使 用 那些 用 到 invokedynamic 的 语言 和 框架 不 一 定 要 完全 
搞 清 楚 这 些 内 容 。 


5.5.1 invokedynamic 如 何 工 作 
为 了 支持 invokedynamic，Java 7 又 新 增加 了 几 条 常量 池 定 义 。 这 些 是 在 Java 6 技术 中 无 法 
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提供 的 支持 。 

给 invokedynamic 指 邻 的 索引 必须 指向 类 型 为 CONSTANT_InvokeDynamic 的 常量 , 这 个 常 
量 上 是 两 个 16 位 的 索引 (也 就 是 4 字 节 ) 第 一 个 索引 指向 方法 表 (用 来 确定 要 调用 什么 ) 它们 
被 称 为 引导 方法 (有 时 简写 为 BSM )， 并 且 必 须 是 静态 的 ， 还 要 有 确定 的 参数 签名 。 第 二 个 索引 
指向 CONSTANT_NameAndType。 

从 中 可 以 看 出 CONSTANT_InvokeDynamic 和 普通 的 CONSTANT_MethodRef 差 不 多 ， 只 是 
CONSTANT_MethodRef 指 明 在 哪个 类 的 常量 池 里 找寻 方法 ， 而 invokedynamic 调 用 则 通过 引导 
方法 来 寻找 管 案 。 

引导 方法 会 返回 一 个 cal1Site 实 例 ， 用 它 来 接收 与 调用 点 相关 的 信息 ， 并 连接 动态 调用 。 
调用 点 中 有 一 个 MethoaHandle， 调 用 点 在 这 里 起 一 个 代理 的 作用 ， 对 它 的 所 有 调用 实际 上 就 是 
对 MethodaHandle 的 调用 。? 

invokedynamic 一 开始 并 没有 目标 方法 (还 设 连接 )。 在 第 一 次 调用 时 ， 该 点 的 引导 方法 被 
调用 。 引 导 方 法 返回 一 个 callsite， 它 被 连接 到 ijnvokedynamic 指 令 上 。 该 过 程 如 图 5-5 所 示 。 


Invokevirtual invokedyramic 


boot strapMethod() 


图 5-5 ”虚拟 与 动态 调用 


连接 上 callsite 后 ， 就 可 以 调用 真正 的 方法 了 , 即 callsite 持 有 的 MethodHandle 所 指向 
的 方法 。 这 种 设 定 表明 JIT 编 译 器 可 以 像 优 化 invokevirtual 调 用 那样 优化 invokedvynamic 调 
用 。 下 一 曹 会 讨论 更 多 有 关 优 化 的 内 容 。 

还 有 一 点 值得 注意 ， 革 些 cal1site 对 象 是 可 以 重 连 的 (在 它们 的 生命 期 内 指向 不 同 的 目标 
方法 )。 一 些 动 态 语言 会 大 量 使 用 这 一 特性 。 

下 一 节 会 给 出 一 个 简单 的 例子 ， 我 们 可 以 看 到 invokedynamic 调 用 在 字 节 码 中 如 何 表 示 。 


5.5.2 示例 : 反 编 译 ijnvokedynamic 调用 


如 前 所 述 ，Java 7 中 没有 文 持 invokedynamic 的 Java 语 法 。 要 得 到 带 有 动态 调用 指令 的 .class 
文件 ， 你 内 能 向 字 节 码 处 理 类 库 求 助 。ASM 类 库 ( http://asm.ow2.org/ ) 就 是 一 个 不 错 的 选择 一 一 
它 是 一 个 工业 级 类 库 ， 在 Java 框 架 中 得 到 了 广泛 应 用 。 


DD 详情 请 参见 callsite 的 Javadoc: http.J/cropenjdk .java.net/~jrose/pres/indy-javadoc-mivm/java/lang/invol 
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.4 invoked' EF; 


我 们 可 以 用 这 个 类 库 构 造 一 个 包含 invokedynamic 指 令 的 类 ， 然 后 将 其 转换 为 字 节 流 。 这 
既 可 以 写 到 磁盘 里 ， 也 可 以 交 给 类 加 载 器 捕 人 到 运行 的 VM 中 。 
一 个 简单 的 例子 是 让 ASM 产 生 的 类 包含 一 种 invokedynamic 指 令 的 静态 方法 。 这 个 方法 可 
以 由 普通 的 Java 代 码 调用 一 一 它 封装 ( 或 隐藏 ) 了 真正 调用 的 动态 本 质 。 作 为 invokedynamic 
开发 工作 的 一 部 分 , Remi Forax 和 ASM 团 队 提供 了 一 个 简单 的 工具 来 产生 这 样 的 测试 类 。ASM 是 
第 一 批 完 全 文 持 新 字 节 码 的 工具 之 一 。 
让 我 们 来 看 一 下 这 种 封 骤 方法 的 字 世 码 ; 
public static java.math.BigDecimal invokedynamic|(); 
Code: 
0: invokedynamic #22, 0 
=“ // InvokeDynamic #0: :(})Ljava/math/BigDecimal:; 
5 areturn 
到 目前 为 止 还 没什么 看 头 , 因为 复 厅 性 主要 体现 在 常量 池 中 ,我们 来 看 看 和 动态 调用 相关 的 
常量 池 条 目 : 
RootstrapMethods: 
0; #17 invokeastatic test/invdyn/DynamicIndyMakerMain .bam: 
“hb (Lijava/lang/invoke/MethodHandleas$Lookup;Ljava/lang/string,; 
ss Ljava/lang/invoke/MethodType;Ljava/lang/Qbject;) 
= Ljavallang/invoke,/Callsite; 
Method arguments.: 
#19 1234567890 .1234567890 


#10 = Utf (JjLiava/math/BigDecimal:; 

#18 = Utf8a 234356 T7890 .1234567890 

村 19 三 String 拉 1 和 日 i 1l234567890.1234567890 

#20 = Utf8 

#21] = NameAndType 持 立 口 : 排 工 间 /1 :(}lLjava/math/BigDecimal,; 
#22 = InvokaeDynamic 林 各 :性 号 工 :i #0: :()Lijava/math,/BigDecimal 


要 想 完全 搞 清 楚 确 实 得 花 点 心思 琢磨 琢磨 。 我 们 逐一 来 看 一 下 。 

口 invokedynamic 操 作 公 在 条 目 松 2 中 。 它 指向 引导 方法 0 和 NameAndType#21。 

口 在 可 的 BSM 是 类 DynamicIndyMakerMain 中 的 普通 静态 方法 bsm() 。 它 有 BSM 的 正确 签名 。 

口 条 目 松 1 给 出 了 这 个 动态 连接 点 的 名 称 “”“， 还 有 返回 类 型 BigDecimal (保存 在 #10 )。 

口 条 目 机 9 是 传人 引导 方法 的 静态 参数 。 

如 你 所 见 , 这 里 需要 做 很 多 基础 工作 来 保证 类 型 安全 ,但 在 运行 时 出 错 的 方式 仍然 还 有 很 多 ， 
但 这 种 机 制作 了 很 大 贡献 ， 它 在 保留 了 灵活 性 的 同时 提供 了 安全 性 。 


注意 BootstrapMethods 方 法 指向 方法 句柄 而 不 古 直 接 指 向 方法 ， 这 提供 了 额外 的 间接 性 ， 
或 者 说 灵活 性 。 在 前 面 的 讨论 中 我 们 并 没有 涉及 ， 因 为 它 可 能 会 混淆 正在 发 生 的 事情 ， 
对 于 理解 这 种 机 制 如 何 工 作 并 没有 实质 性 的 帮助 。 


到 此 为 止 ， 我 们 已 经 结束 了 对 invokedynamic 和 字 节 和 


及 类 加 载 内 部 工作 机 制 的 讨论 。 
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5.6 小结 


在 本 章 中 , 我 们 快速 浏览 了 字 节 码 和 类 加 载 ， 还 解剖 了 类 文件 ,并 简单 介绍 了 JVM 提 供 的 运 
行 时 环境 。 随 着 对 平台 内 部 更 深入 地 了 解 ， 我 们 相信 你 会 成 为 更 厉害 的 开发 者 。 

希望 你 从 本 章 中 学 到 如 下 知识 : 

口 类 文件 格式 和 类 加 载 是 JVM 操 作 的 核心 ,它们 对 于 任何 想 在 VM 上 运行 的 语言 来 说 十 分 重要 ; 

口 类 加 载 的 各 个 阶段 同时 保证 了 运行 时 的 安全 和 性 能 特性 ; 

口 方法 句柄 是 Java 7 主要 新 API 之 一 ， 它 是 反射 之 外 的 一 个 可 选 方案 ; 

口 JVM 按 相关 功能 分 为 不 同 的 族 系 ; 

口 Java 75 引 人 invokedvnamic: 一 种 调用 方法 的 新 办 法 。 

现在 是 时 候 进 入 下 一 个 大 主题 了。 通过 阅读 下 一 章 ， 你 会 在 性 能 分 析 方 面 打下 坚实 的 基础 。 
你 将 学 会 如 何 评估 和 优化 性 能 以 及 如 何 充 分 发 挥 JVM 核 心 技术 的 能 力 ( 比如 JIT 编 译 嚣 ， 它 会 把 
字 节 码 转换 成 超 快 的 机 器 码 )。 
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本 章 内 容 

口 性 能 的 重要 性 

口 新 的 垃圾 收集 器 Gl 

口 VisualVM: 内 存 可 视 化 工具 
口 即时 编译 


精 粒 的 性 能 会 “ 杀 死 ”你 的 应 用 程序 ， 使 你 名 声 扫 地 ， 在 客户 中 的 信誉 大 受 影 响 。 除 非 你 具 
有 绝对 昔 断 地 位 ,否则 你 的 客户 将 夺 门 而 出 , 直 奔 你 的 范 争 对 手 而 去 。 要 让 粳 粒 的 性 能 不 再 精 踢 
你 的 项 目 ， 你 需要 理解 性 能 分 析 ， 还 要 知道 如 何 利 用 它 。 

性 能 分 析 与 调 优 是 个 非常 庞大 的 课题 , 但 是 现在 有 太 多 处 理 方式 都 在 误 人 子弟 。 所 以 我 们 准 
备 把 性 能 调 优 的 秘诀 透漏 给 你 。 

秘诀 来 了 一 一 性 能 调 优 唯一 的 惊天 秘诀 就 是 你 必须 量体裁衣 。 没 有 评测 ， 就 没有 合适 的 
调 优 。 

原因 ， 人们 总 是 狂 不 对 系统 变 慢 的 是 哪里 。 所 有 人 都 猜 不 对 。 你 ， 我 ， 甚 至 是 James Gosling 
大 神 一 一 我 们 总 会 心 生 偏见 ， 并 倾向 于 那些 可 能 根本 不 存在 的 模式 。 

实际 上 ,“ 我 的 哪些 Java 代 码 需 要 优化 ?” ”这 个 问题 的 答案 经 党 是 “哪个 也 不 用 , 都 挺 好 的 ”。 

假设 有 一 个 经 典 ( 相当 保守 ) 的 电子 商务 Web 应 用 为 注册 客户 提供 服务 。 它 有 一 个 SQL 数据 
库 ， 一 个 面向 Java 应 用 服务 器 的 Apache Web 服 务 器 ， 以 及 连接 这 一 切 的 标准 网 络 配置 。 系 统 真正 
的 瓶颈 经 常 是 非 Java 部 分 ( 数据 库 、 文 件 系统 、 网 络 ), 但 不 经 过 评测 ，Java 开 发 人 员 永 远 都 不 会 
知道 问题 出 在 哪里 。 开 发 人 员 不 去 解决 真正 的 问题 ,而 是 把 时 间 廊 费 在 对 于 改进 系统 性 能 训 无 意 
义 的 代码 微调 上 。 

你 希望 能 够 回答 如 下 几 类 基本 问题 。 

口 如 果 你 们 搞 了 次 促销 ， 客 户 突然 暴 增 十 售 ， 系 统 有 足够 的 内 存 来 应 付 这 种 局 面 吗 ? 

口 客户 从 应 用 程序 中 看 到 的 平均 响应 时 间 是 多 长 ? 

口 跟 竞争 对 手 比 起 来 怎么 样 ? 

要 做 性 能 调 优 , 你 就 不 能 猜测 导致 系统 变 慢 的 厚 因 。 你 必须 知道 并 且 确 保 你 真正 实现 性 能 调 
优 的 唯一 办 法 就 是 性 能 评测 。 
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你 还 需要 明白 性 能 调 优 不 是 ， 

口 一 堆 技 巧 和 窒 门 ; 

口 秘 窗 武 毅 ; 

口 你 在 项 目 结束 时 撒 一 把 的 仙 粉 。 

对 “技巧 和 窗 门 ”要 特别 小 心 。JVM 是 一 个 非常 复杂 ， 并 经 过 高 度 优化 的 环境 ， 如 果 脱 离 了 
上 下 文 , 这 些 技巧 基本 都 设 什么 用 ,而 且 可 能 还 会 带 来 麻烦 。 随 着 JVM 在 代码 优化 方面 越 来 越 智 
能 ， 它 们 也 很 快 就 会 过 时 。 

性 能 分 析 实 际 上 是 种 试验 性 的 科学 。 你 可 以 把 代码 看 成 是 某 种 科学 试验 , 有 输入 , 会 产生 “ 输 

标 表明 系统 执行 任务 的 效率 。 性 能 工程 师 的 工作 是 研究 这 些 输出 ， 并 找 出 其 中 的 
模式 。 所 以 性 能 调 优 是 统计 应 用 的 一 个 分 支 ， 而 不 是 一 群 老太婆 的 用 言 碎 语 。 

本 章 将 是 你 的 新 起 点 , 我 们 会 向 你 介绍 Java 性 能 调 优 实战 。 但 这 是 个 大 课题 , 由 于 篇 幅 有 限 ， 
我 们 只 能 把 你 领 进门 ,帮助 你 分 析 其 中 的 重要 原理 和 标志 性 内 容 。 我 们 也 会 尽量 解答 大 部 分 基本 
问题 。 

口 性 能 为 什么 这 么 重要 ? 

ep 

| 会 让 调 优 变 得 复杂 ? 
0 频次 扣 得 老虎 和 守 并 竹 攀 油 优 。 
口 哪些 是 导致 系统 迟缓 的 最 常见 原因 ? 
我 们 还 会 介绍 JVM 中 与 性 能 相关 的 两 个 最 重要 子 系统 : 
口 垃圾 收集 子 系统 
口 JIT 编 译 器 
有 了 了 这些， 你 就 可 以 开始 着 手 解决 编码 时 过 到 的 实际 问题 了 。 
我 们 先 来 快速 浏览 一 些 基 本 词汇 ， 以 便 你 可 以 表达 并 框 定 自己 的 性 能 问题 和 目标 。 


6.1 ”性 能 术语 


为 了 让 你 充分 理解 本 章 所 讨论 的 内 容 , 我 们 会 给 出 正规 的 性 能 概念 定义 。 下面 是 性 能 工程 师 
词典 里 最 重要 的 一 些 术 语 : 

口 等 待 时 间 ( Latency ) 

口 吞吐 量 ( Throughpnut ) 

口 利用 率 ( Utilization ) 

口 效率 ( Efficiency ) 

口 容量 ( Capacity ) 

口 扩展 性 (Scalability ) 

口 退化 (Degradation ) 

Doug Lea 讨 论 这 些 术 语 时 都 是 放 在 多 线程 代码 的 上 下 文中 , 但 我 们 要 考虑 的 范围 更 广 : 从 一 
个 多 线程 处 理 器 到 整个 集群 服务 器 平台 。 
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6.1.1 ”等待 时 间 


等 待 时 间 是 在 给 定 工作 量 下 处 理 一 个 任务 单元 所 消耗 的 时 长 。 通 常 ， 都 是 在 工作 量 “ 正 常 ” 
的 情况 下 提 到 等 待 时 间 的 ,但 有 价值 的 性 能 评测 一 般 都 是 用 一 张 图 形 来 显示 在 工作 量 不 断 增 加 的 
情况 下 等 待 时 间 随 之 改变 的 函数 关系 。 

图 6-1 显 示 了 在 工作 量 增加 时 ， 某 一 性 能 指标 ( 比如 等 待 时 间 ) 出 现 了 一 个 突 发 的 非 线 性 退 
化 。 这 通常 被 称 为 性 能 肘 。 


70 40 .0 12.0 16.0 
侦 载 


图 6-1 性 能 肘 


6.1.2 ”吞吐 量 

吞吐 量 是 系统 在 限定 资源 、 限定 时 长 内 能 完成 的 单位 工作 量 。 用 的 最 多 的 是 在 某 一 参考 平台 
( 比如 指明 了 硬件 配置 、 操 作 系 统 和 软件 环境 的 特定 品牌 服务 器 ) 上 的 每 秒 事务 处 理 数 。 
6.1.3 ”利用 率 


利用 率 表示 可 用 资源 中 用 来 处 理工 作 单 元 ( 而 不 是 清理 任务 或 处 于 空闲 状态 ) 的 资源 百分比 。 
人 们 通常 会 说 服务 器 的 利用 率 是 10%， 这 其 实 是 说 在 正常 处 理 时 间 内 处 理工 作 单 元 的 CPU 百 分 
比 。 注 意 ， 不 同 资源 的 利用 率 水 平 可 能 有 非常 大 的 差异 ， 比 如 CPU 和 内 存 之 间 。 


6.1.4 ”效率 
改 率 更 差 


系统 的 效率 等 于 在 吐 量 除 以 所 用 资源 。 一 个 用 更 多 资源 产生 相同 吞吐 量 的 系统 交 
比如 比较 两 个 集群 方案 。 如 果 为 了 达到 相同 的 吞吐 量 ， 方案 A 需 要 的 服务 器 数量 是 方案 B 的 
两 售 ， 则 方案 A 的 效率 是 B 的 一 半 。 
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别 忘 了 ， 资 源 也 可 以 用 成 本 来 衡量 一 一 如 果 方 案 X 的 成 本 是 方案 Y 的 两 信 或 需要 两 全 的 员工 
运行 生产 环境 ， 则 方案 X 的 效率 是 Y 的 一 半 。 


6.1.5 ”容量 
容量 是 任 一 时 刻 能 通过 系统 的 工作 单元 ( 比如 事务 ) 数量 。 也 就 是 在 特定 的 等 行 时 间或 夺 吐 
量 下 ， 能 够 得 到 同步 处 理 的 工作 单元 数量 。 


当 系 统 得 到 更 多 资源 时 , 它 的 奉 吐 量 或 等 待 时 间 会 发 生变 化 。 这 种 发 生 在 奉 吐 量 或 等 待 时 间 
上 的 变化 就 是 系统 的 扩展 性 。 

如 果 方 案 A 可 用 的 服务 状 数 量 翻 倍 ， 它 的 吞吐 量 也 能 翻 倍 ， 那 我 们 就 说 它 实 现 了 完美 的 线性 
展 。 在 大 多 数 情况 下 ， 和 拖 美 的 线性 扩展 很 难 达到 。 

还 应 该 注意 ,系统 的 扩展 性 取决 于 很 多 因素 ,而 且 这 种 扩展 性 还 是 变化 的 。 系 统 能 以 线性 方 
趟 巾 上 扩展 到 某 一 点 ， 然 后 开始 退化 。 这 是 为 外 一 种 性 能 肘 。 


6.1.7 退化 


如 果 在 不 增加 资源 的 情况 下 增加 工作 单元 或 网 络 系统 的 客户 端 , 一 般 等 待 时 间或 吞吐 量 都 会 
发 生变 化 。 这 是 系统 在 负载 增加 时 出 现 的 退化 。 
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和 : a 上 过 儿 刁 次 间 直 和 ph ja a 是 i 和 : 
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rg 还 化 是 负面 的 也 就 是 说 给 系统 增加 工作 单元 会 对 性 能 产生 负面 影响， 此 
如 导致 处 理 等 待 时 间 变 长 。 但 某 些 情况 下 退化 也 有 可 能 是 正面 的 。 

比如 说 ， 如 果 过 重 的 负 栽 导致 系统 某 些 部 分 超过 了 冰 值 ， 迫 使 系统 切换 到 高 性 能 模式 ， 这 
会 让 系统 工作 效率 更 高 ， 缩 短处 理 时 间 ， 尽 管 还 要 完成 更 多 工作 。JVM 是 个 动态 性 非常 强 的 运 
行 时 系统 ， 并 且 有 几 部 分 可 以 达成 这 种 效果 。 


i 


前 面 这 些 术语 是 最 常用 的 性 能 指标 ,当然 还 有 其 他 一 些 重要 的 指标 , 但 这 些 是 指导 系统 性 能 
凋 优 的 基本 统计 数据 。 在 下 一 节 中 , 我 们 会 给 出 一 个 以 密切 关注 这 些 数值 为 基础 ， 并 尽 可 能 定量 
的 性 能 调 优 方法 。 
6.2 务实 的 性 能 分 析 法 

许多 开发 人 员 在 接 到 性 能 分 析 任务 时 , 脑子 里 都 不 清楚 他 们 要 通过 分 析 得 到 什么 。 所 有 开发 
人 员 或 经 理 在 开始 做 这 件 事 时 经 常 只 是 模 模糊 糊 地 感觉 代码 “应 该 跑 得 更 快 ”。 


但 这 是 彻底 的 倒退 。 要 进行 真正 有 效 的 性 能 调 优 ,在 开始 做 任何 技术 类 工作 之 前 ,你 应 该 先 
认真 考虑 下 面 这 些 问题 并 找 出 答案 。 


各 
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口 你 正在 测量 的 代码 有 哪些 可 观测 的 环节 ? 

口 如 何 测 量 那 些 可 观测 环节 ? 

口 这 些 可 观测 环节 的 目标 是 什么 ? 

口 你 怎么 判断 性 能 调 优 是 天 做 好 了 ? 

口 性 能 调 优 可 接受 的 最 大 支出 是 多 少 ( 按 开 发 人 员 投入 的 时 间 和 增加 的 代码 复杂 度 计算 )? 

口 在 优化 的 过 程 中 ， 哪 些 东 西 是 你 不 能 舍弃 的 ? 

最 重要 的 , 也 是 我 们 要 反复 强调 的 ， 就 是 你 必须 测量 。 你 至 少 得 测量 一 个 可 观测 环节 ， 才 算 
得 上 是 在 做 性 能 分 析 。 

当 你 开始 测量 代码 , 便 经 常会 发 现 事情 并 非 你 想 的 那样 。 很 多 性 能 问题 的 根源 可 能 是 一 个 丢 
失 的 数据 库 索 引 ， 或 者 有 争议 的 文件 系统 锁 。 在 优化 代码 时 ， 你 应 该 时 刻 牢 记 代 码 很 可 能 不 是 问 
题 的 关键 。 为 了 定量 分 析 问题 ， 你 首先 需要 知道 目 己 在 测量 什么 。 


6.2.1 知道 你 在 测量 什么 


做 性 能 调 优 必须 测量 一 些 东 西 。 如 果 你 没有 测量 可 观测 环节 ， 就 不 能 算 做 性 能 调 优 。 坐 在 那 


里 有 盯 着 代码 ,希望 脑子 里 勋 出 一 个 可 以 更 快 解决 问题 的 方法 ， 这 可 不 是 性 能 分 析 。 


提示 。 要 成 为 优秀 的 性 能 工程 是， 你 必须 知道 平均 数 、 中 位 数 、 模 式 、 方 差 、 百 分 位 数 、 标 准 
差 、 样 本 大 小 、 正 态 分 布 等 这 样 一 些 术语 。 如 果 还 不 融入 这 些 概念 ， 最 好 现在 就 到 网 上 
搜 搜 ， 如 果 有 必要 的 话 ， 认 真 看 看 搜 出 来 的 内 容 。 


做 性 能 分 析 最 重要 的 是 知道 哪个 可 观测 环节 (上 节 介 绍 的 ) 最 重要 。 你 应 该 总 是 把 测量 结果 、 
目标 和 结论 跟 一 个 或 多 个 基本 可 观测 环节 结合 起 来 。 

这 里 有 些 常见 的 可 观测 项 ， 都 是 性 能 凋 优 的 好 对 象 。 

吕 方法 handleRequest () 运 行 所 需 的 平均 时 间 ( 启动 完成 之 后 )。 

口 并 发 客户 端 数量 为 10 时 ， 系 统 等 待 时 间 的 第 90 个 百 分 位 数 。 

口 把 并 发 用 户 数 从 1 增长 到 1000 时 ， 啊 应 时 间 的 退化 。 

以 上 这 些 都 是 工程 师 想 要 测量 的 代表 性 数值 , 并 很 有 可 能 需要 优化 。 想 得 到 准确 又 有 用 的 数 
值 ， 必 须 掌 握 基 本 的 统计 学 知识 。 

知道 你 要 测量 什么 ,对 数值 的 准确 性 有 信心 是 性 能 调 优 的 第 一 步 。 但 模糊 或 随意 的 目标 通常 
没什么 好 结果 ， 人 性 能 调 优 也 是 如 此 。 


6.2.2 ”知道 怎么 测量 


要 精确 确定 一 个 方法 或 其 他 代码 片段 运行 需要 多 长 时 间 ， 只 有 两 种 方法 : 
口 直接 测量 ， 在 类 源码 中 插入 测量 代码 ; 
口 在 类 加 载 时 把 类 转换 成 受 测 类 。 
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大 多 数 简单 直接 的 性 能 测量 技术 都 依赖 于 以 上 其 中 一 种 或 全 部 技术 。 

还 应 该 提 一 下 JVM 工 具 接 口 (JVMTI )， 用 它 可 以 创建 非常 复杂 的 分 析 器 ， 但 它 也 有 缺陷 。 
它 需 要 性 能 工程 师 编 写本 地 代码 , 并 且 它 产生 的 分 析 数 值 本 质 上 是 统计 平均 值 , 而 不 是 直接 测量 
结果 。 

直接 测量 


直接 测量 是 最 容易 理解 的 技术 ,但 它 是 侵入 式 的 。 最 简单 的 是 像 下 面 这 种 形式 : 


long t0 = System.currentTimeMillis'(}; 

methodToBeMeasured |(); 

long t1 = System.currentTimeMillis(); 

long elapased = tl1 -= t0; 

svatem.out .println(l"methodToBeMeasured took "+ elapsed +" millis"), 


这 段 代 码 会 输出 methodToBeMeasured1l) 精 确 到 芝 秒 的 运行 时 长 。 很 不 方便 的 是 ， 你 机 到 
处 添加 这 种 代码 ， 而 且 随 着 测量 结果 不 断 增多 ， 代 码 很 容易 被 数据 淹没 。 

除 此 之 外 还 有 其 他 问题 ， 如 果 methodToBeMeasured |) 运行 时 长 不 中 一 毫秒 会 出 现 什么 情 
况 ? 稍 后 我 们 就 会 看 到 ， 此 外 还 值得 注意 的 是 冷 启动 效果 一 一 后 运行 的 方法 可 能 比 先 运行 的 快 。 

通过 类 加 载 自动 测量 

我 们 在 第 1 章 和 第 5 章 讨论 过 如 何 把 类 编译 成 可 执行 程序 ,其 中 一 个 关键 步骤 是 在 加 载 字 节 码 
时 进行 转换 。 这 个 特性 非常 强大 ， 是 很 多 现代 Java 平 台 的 核心 技术 。 其 中 一 个 简单 的 例子 就 是 方 
法 的 自动 测量 。 

在 这 种 方法 中 ， 特 殊 的 类 加 载 器 加 载 methodToBeMeasured() 所 属 类 ， 在 方法 开始 和 结束 
的 地 方 加 上 记录 方法 进入 和 退出 时 间 的 字 节 码 。 这些 时 间 通 常会 被 写 入 共享 的 数据 结构 ， 由 其 他 
线程 访问 。 这 些 线程 一 般 会 将 数据 写 人 日 志文 件 , 或 者 通过 网 络 交 给 负责 处 理 原始 数据 的 服务 器 。 

很 多 高 端的 性 能 监测 工具 ( 比如 OpTier CoreFirst ) 都 是 以 这 项 技术 为 核心 的 。 但 在 编写 本 书 
时 ， 这 个 市 场 上 似乎 还 没有 开源 工具 。 


注意 我 们 会 在 后 而 讨论 到 ，Java 方法 开始 时 需要 进行 解释 ， 然 后 才 切 换 到 编译 模式 。 要 得 到 
真正 的 性 能 指标 结果 ， 你 必须 去 掉 解 释 模 式 占用 的 时 间 ， 因 为 它们 会 严重 扭曲 真实 结果 。 
后 面 还 会 给 出 更 多 细节 ， 告 诉 你 如 何 确定 方法 切换 为 编译 模式 的 时 间 。 


你 可 以 用 这 两 项 技术 ( 其 一 或 全 部 ) 找 出 某 一 方法 执行 所 需 的 时 长 。 下 一 个 问题 ， 完 成 调 优 
之 后 ， 你 想得到 什么 样 的 数值 ? 
6.2.3 ”知道 性 能 目标 是 什么 


清晰 的 目标 能 让 人 注意 力 集中 ,所 以 了 解 和 传达 优化 的 最 终 目标 ( 知道 要 测量 什么 ) 至 关 重 
要 。 大 多 数 情 况 下 ， 这 个 目标 简单 而 明确 ， 比 如 : 

口 将 10 个 并 发 用 户 的 问 到 并 等 等 时 间 的 第 90 个 百 分 位 数 减 少 20%，; 

口 将 handleRequest () 的 平均 等 待 时 间 减 少 40%， 方 差 减 少 25%。 
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在 一 些 更 复杂 的 情况 中 ， 目标 可 能 由 几 个 相关 的 性 能 目标 共同 构成 。 你 要 知道 , 你 所 测量 和 
想 要 优化 的 独立 可 观测 项 越 多 , 调 优 工作 就 会 变 得 越 复杂 。 优化 一 个 性 能 目标 可 能 会 对 其 他 性 能 
目标 产生 负面 影响 。 
有 了 时, 在 设 定 目标 之 前 你 很 有 必要 做 些 初步 分 析 ， 比 如 在 确定 要 让 方法 运行 得 更 快 这 一 目标 
之 前 , 应 该 先 确定 哪些 方法 最 重要 。 这 很 好 , 但 经 过 初步 探索 后 , 你 最 好 停 下 来 再 确认 一 下 目标 ， 
然后 再 达成 它们 。 开 发 人 员 非 常 爱 犯 只 顾 低头 拉 车 ， 不 顾 抬 头 看 路 的 错误 。 


6.2.4 ”知道 什么 时 候 停止 优化 


理论 上 来 说 , 知道 什么 时 候 集 止 优化 并 不 难 一 一 达成 目标 之 时 就 是 任务 完成 之 日 。 然 而 实际 
中 人 人 们 很 容易 陷入 性 能 调 优 的 泥 淹 。 如 果 事情 进展 顺利 ， 你 肯定 想 要 继续 前 进 并 做 得 更 好 。 而 如 
末 不 太 有 顺利 ， 你 为 了 达成 目标 就 会 不 断 尝 试 新 事略 。 

要 想 知 道 什么 时 候 停 止 优化 , 你 需要 对 目标 有 清醒 的 认识 并 理解 它们 的 价值 。 能 达成 性 能 目 
标的 90% 通 党 就 足够 ， 你 还 可 以 利用 节省 下 来 的 时 间 去 做 些 别 的 事 。 

还 要 考虑 一 点 ,你 要 看 看 有 和 多少 工 作 投入 到 了 极 少 用 到 的 代码 路 径 上 。 通过 优化 代码 来 减少 
程序 运行 时 长 的 1% ( 甚至 更 少 ) 完全 是 在 浪费 时 间 ， 但 奇怪 的 是 做 这 种 事 儿 的 开发 人 员 数 量 
惊人 。 

至 于 该 优化 什么 , 这 里 有 一 组 非常 简单 的 指导 规则 。 你 可 能 需要 根据 自身 情况 进行 调整 , 但 
它们 的 适用 范围 很 广泛 : 

口 优化 那些 重要 ， 而 不 是 最 容易 的 代码 。 

口 首先 优化 那些 最 重要 ( 通常 是 调用 最 频繁 ) 的 方法 。 

口 在 过 到 那些 唾 手 可 得 的 优化 时 ， 把 它 办 了 , 但 要 清楚 代码 的 调用 频率 。 

最 后 再 做 一 轮 测 量 工作 。 如 果 还 没 达成 性 能 目标 ， 你 就 需要 清查 一 下 ,看 看 离 命中 目标 还 有 
多 大 差距 ， 以 及 取得 的 成 绩 是 不 是 已 经 对 整体 性 能 产生 了 你 所 期 望 的 影响 。 


6.2.5 ”知道 高 性 能 的 成 本 


所 有 性 能 调整 都 贴 着 价 签 。 
口 分 析 和 优化 代码 要 占用 的 时 间 (在 任何 软件 项 目 中 ,开发 人 员 的 时 间 基 本 都 是 最 大 的 开 
这 由 
口 所 做 的 调整 可 能 会 引信 额 外 的 技术 复 灯 度 ( 也 有 简化 代码 的 性 能 优化 , 但 它们 不 是 主流 )。 
口 为 了 让 主 处 理 线 程 运 行 得 更 快 ， 可 能 会 引入 额外 的 线程 来 执行 辅助 任务 ， 但 这 些 线程 可 
能 会 在 负载 较 高 时 对 系统 整体 产生 不 可 预料 的 影响 。 
不 管 是 什么 价 签 ， 你 都 要 重视 ， 并 尽量 在 完成 第 一 轮 优 化 之 前 找到 它们 。 
这 有 助 于 你 了 解 提 高 性 能 的 最 大 可 接受 成 本 。 这 个 成 本 可 能 是 设 定 开发 人 员 调 优 的 时 间 限 
制 ， 额外 的 类 数 或 代码 行 数 。 比 如 说 ,开发 人 员 决 定 花 在 优化 上 的 时 间 不 能 超过 一 个 星期 , 或 者 
因 优 化 而 生 的 类 增长 不 应 该 超过 100%( 即 大 小 变 成 原来 的 两 信 ); 
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6.2.6” 知 这 过 早 优 化 的 危险 
关于 优化 ，Donald Knuth 有 段 著名 的 评论 : 
程序 员 浪 费 了 大 量 时 间 者 虚 , 或 担心 程序 中 无 闫 紧要 部 分 的 达 度 ,并 且 那 些 党 试 改 
进 效率 的 行为 实际 上 有 很 强 的 负面 影响 …… 过 时 全 化 是 万 恶 之 源 。” 
这 段 话 在 业内 引起 了 广泛 争论 ， 而 且 人 们 通常 只 记 住 了 最 后 一 句 。 这 之 所 以 令 人 感到 遗憾 ， 
有 如 下 原因 。 
口 在 评论 的 前 段 ，Knuth 含 蒂 地 提醒 我 们 要 测量 ， 没 有 测量 就 不 能 确定 程序 的 关键 部 分 。 
口 我 们 再 次 提醒 你 ， 可 能 不 是 代码 导致 等 每 时 间 过 长 一 一 环境 中 的 其 他 部 分 也 会 产生 等 待 
时 | 间 。 
口 在 完整 的 评论 中 ， 很 容易 看 出 Knuth 是 在 谈论 那些 有 意识 的 、 齐 心 协 力 的 优化 。 
口 这 段 评论 的 简短 版 让 它 变 成 了 不 良 设计 或 精 粒 执行 选择 的 相当 巧合 的 借口 。 
有 些 优 化 体现 在 良好 的 编码 风格 上 : 
口 不 要 分 配 不 需要 的 对 象 。 
口 如 果 再 也 不 需要 调试 日 志 ， 就 去 掉 它 。 
我 们 在 下 面 的 代码 中 加 了 一 个 检查 , 看 日 志 对 象 是 否 处 理 调试 日 志 。 这 种 检查 被 称 为 日 志和 守 
卫 。 如 有 果 日 志 了 于 系统 被 设置 为 不 处 理 调 试 日 志 ， 这 段 代 码 就 不 会 构造 日 志 消 息 ， 省 掉 了 为 了 日 志 
消 昌 而 调用 currentTimeMillis() 和 构造 stringBuilaer 对 象 的 开销 。 


if (log.iaDebugEnabled()) log.debugl"Useless log at: "+ 
syastem. currentTimeMillis()):; 


但 如 果 调 试 日 志 真 的 没有 用 ,我 们 可 以 把 这 段 代 码 一 并 去 掉 , 就 能 再 节省 两 个 处 理 器 周期 (日 
志 守 卫 的 开销 )。 

性 能 调 优 的 工作 之 一 就 是 从 一 开始 就 写 出 质地 优良 、 高效 运行 的 代码 ,更 好 地 认识 Java 平 台 ， 

的 底层 运行 机 制 ( 比如 理解 在 合并 两 个 字符 串 时 隐 含 的 对 象 分 配 )， 并 在 编码 时 考虑 到 性 
能 问题 ， 才 能 写 出 更 好 的 代码 。 

现在 我 们 有 了 框 定性 能 问题 和 目标 的 基本 词汇 , 还 有 如 何 解决 问题 的 方法 大 纲 。 但 我 们 还 没 
解释 为 什么 这 是 软件 工程 师 会 遇 到 的 问题 ， 以 及 这 种 需求 来 自 哪里 。 要 弄 懂 这 个 , 我们 有 必要 简 
单 了 解 一 下 硬件 的 世界 。 


6.3 哪里 出 错 了 ? 我 们 担心 的 原因 


在 几 年 前 , 性 能 问题 看 起 来 并 不 重要 。 时钟 速度 不 断 上 升 , 所 有 软件 工程 师 只 要 多 等 几 个 月 ， 
哪 介 征 与 得 很 位 的 代码 ， 也 能 倩 助 节 闻 攀 升 的 CPU 速度 表现 出 优异 性 能 。 


中 Donald E. Knuth,“ 带 go to 语句 的 结构 化 编程 ”， 计 算 调查 ，6，no.4 ( 1974 年 12 月 )。http://pplab.snu.ac.kricourses/ 
adv_pl0S/papers/p261-knuth pdf , 
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那么 ， 事 情 怎 么 会 错 得 这 么 离谱 ? 为 什么 时 钟 速度 的 提升 不 再 那么 快 了 ? 更 让 人 担忧 的 是 ， 
为 什么 有 3GHz 世 片 的 电脑 看 起 来 比 2GHz 芯 片 的 快 不 了 多 少 ” 行业 中 软件 工程 师 需要 考虑 性 能 
问题 的 这 种 趋势 是 从 哪里 来 的 ? 

我 们 会 在 本 节 讨 论 引 导 这 股 趋势 的 力量 , 以 及 连 最 纯粹 的 软件 开发 人 员 也 需要 了 解 一 点 硬件 
知识 的 原因 。 我 们 会 为 本 章 剩 下 的 内 容 打 好 基础 ， 让 你 理解 JIT 编 译 的 概念 和 一 些 有 深度 的 例子 。 

你 可 能 听 说 过 “摩尔 定律 "。 很 多 开发 人 员 都 知道 这 个 定律 与 提高 计算 机 速度 有 关 ， 但 对 于 
具体 细节 不 其 了 了 。 我 们 来 解释 一 下 它 到 底 是 什么 意思 ， 以 及 它 在 不 久 的 将 来 可 能 带 来 的 影响 。 


6.3.1 摩尔 定律 一 一 过 去 和 未 来 的 性 能 趋势 


摩尔 定律 是 Gordon Moore 提 出 来 的 ， 他 是 Intel 的 创始 人 之 一 。 该 定律 最 常见 的 形式 之 一 是 : 
晶片 上 的 晶体 管 数量 每 两 年 翻 一 香 是 合算 的 。 
这 个 定律 实际 上 是 对 计算 机 处 理 咒 (CPU ) 发 展 趋势 的 一 种 看 法 ， 基 于 Gordon Moore 在 1965 
年 发 表 的 一 篇 论文 ,最初 是 对 未 来 10 年 进行 预测 一 也 就 是 直到 1975 年 。 而 这 一 预测 直到 现在 仍 
然 能 够 应 验 ( 据 预 测 其 在 2015 年 以 前 都 有 效 )， 实 在 值得 称道 。 
我 们 在 图 6-2 中 绘制 了 Intel x86 家 族 从 1980 年 发 展 到 2010 年 的 订 整 个 历程 中 的 一 些 真实 CPU。 
图 中 显示 了 不 同时 期 发 布 的 芯片 所 集成 的 晶体 管 数量 。 
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图 6-2 ”随时 间 变 化 的 晶体 管 数 量 对 数 线性 图 


这 是 一 个 对 数 线性 图 ， 所 以 y 轴 上 的 每 次 增长 都 是 前 一 个 值 购 10 倍 。 如 你 所 见 ， 这 条 线 基本 
是 和 二 的 ， 而 且 每 隅 六 或 七 年 就 会 穿 过 一 个 垂直 层级 。 这 证 明了 摩尔 定律 的 准确 性 ， 因 为 六 或 七 年 
增长 十 倍 和 每 隔 两 年 翻 一 番 基 本 是 一 致 的 。 

图 中 y 轴 的 刻 诬 是 对 数 ， 这 就 是 说 Intel 在 2005 年 生产 的 主流 新 品 大 概 有 一 亿 个 晶体 管 。 这 是 
1990 年 生产 的 芯片 上 的 晶体 管 数量 的 100 倍 。 
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一 定 要 注意 摩尔 定律 特别 谈 到 了 晶体 管 数量 。 要 知道 , 单 赁 摩尔 定律 不 足以 让 软件 工程 师 继 
续 从 硬件 工程 师 那 里 得 到 好 处 ， 理 解 这 一 点 是 最 基本 的 要 求 。 


注意 ”晶体管 数 量 和 时 钟 速度 不 是 一 回 事 儿 ， 人 们 一 般 会 认为 更 高 的 时 钟 速 度 就 意味 着 更 好 的 
性 能 ， 其 实 这 是 一 种 过 于 草率 的 想法 。 
摩尔 定律 在 过 去 很 有 指导 意义 ， 并 且 在 未 来 一 段 时 间 内 应 该 仍然 准确 ( 有 不 同 的 估算 ， 但 在 
2015 年 前 看 起 来 是 合理 的 ) 但 摩尔 定律 计算 的 是 晶体 管 数量 ， 用 这 个 数值 来 指导 开发 人 员 提 高 
代码 性 能 越 来 越 不 靠 谱 。 实 际 上 ， 你 会 发 现 情况 要 更 复杂 。 
在 实际 操作 中 ,性 能 是 由 一 系列 因素 共同 决定 的 , 而 且 每 个 因素 都 很 重要 。 然而 如 果 硬 让 我 
们 从 里 面 挑 一 个 ， 应 该 是 定位 指令 与 数据 运行 速度 的 相关 性 。 这 个 概念 对 性 能 非常 重要 ， 我们 会 
深入 了 解 。 


6.3.2 “理解 内 存 延迟 层级 


计算 机 处 理 器 要 处 理 数据 。 如 果 它 要 处 理 的 数据 到 不 了 了 ,CPU 时 钟 多 快 都 没 用 一 一 它 只 能 等 
着 ,执行 空 操作 ( NOP )， 在 数据 到 来 之 前 基本 就 处 于 停 转 状态 。 

也 就 是 说 在 解决 延迟 时 ， 最 重要 的 两 个 问题 是 , “CPU 核心 要 处 理 的 数据 的 最 近 的 复 本 在 哪 
里 ?“, 还 有 “把 它 送 到 核心 能 用 的 地 方 需要 多 长 时 间 ? ”主要 有 以 下 几 种 答案 。 

口 告 存 器 : 这 是 CPU 上 的 内 存 地 址 ， 随 时 可 用 。 这 部 分 内 存 是 指令 直接 操作 的 。 

口 主 存 : 这 一 般 是 DRAM。 访问 时 间 在 50 纳 秒 左 右 (关于 如 何 使 用 处 理 器 缓存 避免 这 段 延 

迟 请 参见 后 续 内 容 )。 

口 固态 磁盘 ( SSD ): 访问 这 种 磁盘 所 需 的 时 间 不 足 0.1 毫 秒 ， 但 跟 传统 硬盘 比 起 来 ， 它 们 要 

便宜 一 些 。 

口 硬盘 : 访问 这 种 磁盘 并 把 数据 加 载 到 主 存 中 大 概 需 要 5 毫秒 。 

摩尔 定律 预测 了 品 体 管 数量 的 指数 级 增长 , 这 对 内 存 也 有 好 处 一 一 内 存 访问 速度 也 能 以 指数 
级 增长 。 但 这 两 种 指数 并 不 相同 。 内 存 速 度 的 提升 比 CPU 晶 体 管 数量 的 增长 要 慢 得 多， 这 意味 着 
核心 迟早 会 因为 设 有 相关 数据 可 以 处 理 而 藩 人 空闲 状态 。 

为 了 解决 这 个 问题 ， 在 寄存 器 和 主 存 之 间 引 入 了 缓存 。 缓 存 是 少量 更 快 的 内 存 (SRAM， 而 
不 是 DRAM ). 这 种 更 快 的 内 存 成 本 要 比 DRAM 高 很 多 ( 无 论 是 金钱 还 是 晶体 管 )， 所 以 计算 机 不 
全 用 SRAM 作 为 内 存 。 

缓存 分 为 一 级 缓存 (LI ) 和 二 级 缓存 (L2 ) ( 某 些 机 器 还 有 L3 )， 数 值 表 明 缓 存 到 CPU 的 上 臣 
高 ( 越 近 越 快 ) 我 们 会 在 6.6 节 (在 JIT 编 译 上 ) 详细 讨论 缓存 ， 并 给 出 一 个 例子 来 表明 L1 缓 存 
对 运行 代码 的 重要 影响 。 图 6-3 展 示 了 L1 和 L2 组 存 比 主 存 快 多 少 。 之 后 ， 我 们 还 会 给 出 一 个 例子 
来 阐明 这 些 速度 差异 对 运行 代码 的 性 能 有 什么 影响 。 
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图 6-3 ”寄存 化 、 处 理 硕 组 仔 和 主人 存 的 相对 访问 时 间 
除了 增加 缓存 ，20 世 纪 90 年 代 到 21 世 纪 早 期 大 量 使 用 了 另外 一 种 技术 解决 内 存 延 迟 的 问题 ， 


就 是 增加 处 理 器 的 功能 , 这 使 得 处 理 器 越 来 越 复杂 。 即 便 CPU 处 理 能 力 和 内 存 延 迟 之 间 的 差 距 越 
来 武大 ， 也 仍然 采用 复杂 的 硬件 技术 来 保证 CEU 有 数据 可 以 处 理 ， 比 如 指令 级 并 行 ( ILP ) 和 芯 
片 多 线程 (CMT )。 

这 些 技术 的 出 现 消 耗 了 CPU 唱 体 管 预算 中 的 大 部 分 , 并 且 它 们 使 真实 性 能 收益 递减 。 这 一 趋 
势 导致 了 新 观点 的 出 现 ， 即 在 未 来 设计 带 有 多 个 (或 很 多 ) 核心 的 CPU 芯片 。 

这 意味 着 未 来 的 性 能 和 并 发 密切 相关 一 一 主要 办 法 之 一 就 是 通过 拥有 更 多 核心 让 系统 整体 
性 能 得 到 提升 。 那样 ， 即 便 有 一 个 核心 在 等 待 数 据 ， 其 他 核心 也 可 以 继续 工作 。 这 种 关系 十 分 重 
要 ， 所 以 我 们 要 一 再 提起 。 

口 将 来 的 CPU 是 多 核 的 。 

口 性 能 和 并 发 绑 在 一 起 变 成 了 相同 的 关注 点 。 

Java 程 序 员 除 了 要 关注 硬件 ， 还 要 注意 JVM 特 性 带 来 了 额外 的 复杂 性 。 下 一 节 我 们 就 来 看 一 
下 这 些 内 容 。 


6.3.3 ”为 什么 Java 性 能 调 优 存在 困难 


在 JVM 或 其 他 任何 受 控 运行 时 环境 上 做 性 能 调 优 天 生 就 比 在 非 受 控 环 境 下 做 调 优 困难 。 这 是 
因为 CWC++ 程 序 员 几乎 所 有 事情 都 要 自己 做 。0OS 只 提供 很 少 的 服务 ， 比 如 基本 的 线程 调度 。 

在 受 捧 系 统 中 ， 基 本 观点 是 让 运行 时 来 控制 环境 ， 不 用 开发 人 员 目 己 处 理 所 有 细节 。 这 能 提 
高 程序 员 的 生产 率 , 但 要 放弃 革 些 控制 权 。 另 外 一 种 选择 是 放弃 受 控 运行 时 提供 的 所 有 便利 ,但 
和 性 能 调 优 所 做 的 工作 相 比 ， 这 个 代价 实在 太 高 了 。 

造成 调 优 困 难 的 平台 特性 主要 是 : 

口 线程 调度 ; 

口 垃圾 收集 ( GC ); 

口 即时 (JIT ) 编 详 。 

这 些 特性 能 以 很 巧妙 的 方式 交互 。 例 如， 编译 子 系统 用 计时 器 来 决定 编译 哪个 方法 。 也 就 是 
说 等 待 编译 的 候选 方法 集 可 能 会 受到 调度 和 GC 等 特性 的 影 啊 。 每 次 运行 时 所 编译 的 方法 可 能 都 
不 同 。 

正如 你 在 本 节 中 看 到 的 , 准确 测量 是 性 能 分 析 决 策 过 程 的 关键 。 如 果 你 决定 认真 对 待 性 能 调 
优 ， 那 么 理解 Java 平 台中 处 理 时 间 的 细节 和 限制 就 非常 有 用 。 
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6.4 ”一 个 来 自 于 硬件 的 时 间 问 题 


你 有 没有 想 过 计算 机 里 的 时 间 存 在 哪里 以 及 在 哪里 处 理 ? 我 们 都 知道 硬件 最 终 负责 跟 踩 时 
间 ， 但 事实 可 能 不 像 你 想 的 那么 简单 。 

为 了 进行 性 能 调 优 , 你 需要 对 时 间 如 何 工 作 有 深刻 的 认识 。 为 此 我 们 先 从 底层 硬件 开始 讨论 ， 
然后 探讨 Java 如 何 与 这 些 子 系统 集成 ， 最 后 介绍 nanoTime () 方 法 的 复杂 性 。 


6.4.1 硬件 时 钟 


在 基于 x64 的 机 各 里 有 四 种 不 同 的 硬件 时 间 源 ; RTC、8254、TSC 以 及 HPET。 

实时 时 钟 (RTC ) 基本 上 和 便宜 的 电子 表 ( 基于 石英 晶体 ) 里 找到 的 电子 器 件 一 样 ， 在 系统 
断 电 时 由 主板 上 的 电池 供电 。 系 统 在 启动 时 就 是 从 它 那 里 得 到 时 间 的 ， 不 过 很 多 机 器 在 OS 启动 
过 程 中 会 通过 网 络 时 间 协 议 ( Network Time Protocol，NTP ) 跟 网 络 上 的 时 间 服 务 器 同步 。 


ar i = a pe i i 
We i i 国立 F 上 nn | HW a 由 二 "| 
a 可 学 加 | = -二 A 四 [ po y — i 35 = } 


由 而 本 了 中 i 和 站 人 


pry epee 
， 但 现在 它 的 准确 度 对 于 关键 应 用 来 说 已 经 不 够 用 了 。 以 “新 ”或 “ 快 ”命名 的 创新 经 常 是 
这 种 结局 ， 比如 巴黎 的 PontNeuf( “新 桥 " ), 它 建 于 1607 年 ， 现在 已 经 是 书生 市 内 最 古老 的 桥 了 。 


8254 是 可 纺 程 计时 世 片 ， 也 是 始祖 级 的 东西 。 它 的 时 钟 源 是 一 个 119.318kHz 的 晶体 ， 这 个 频 
亩 是 NTSC 彩 色 副 载波 频率 的 三 分 之 一 , 这 也 是 它 返 回 到 CGA 图 形 系统 的 原因 。 它 曾经 为 OS 调 度 
大 提供 定期 时 点 (用 于 时 间 片 )， 但 现在 已 经 有 其 他 时 间 源 (或 者 不 再 需要 ) 了 。 

下 面 介绍 应 用 最 广泛 的 现代 计时 器 一 一 时 间 惟 计 时 器 ( TSC )。 基 本 上 ， 这 是 一 个 跟踪 CPU 
运行 了 多 少 个 周期 的 CPU 计 数 器 。 乍 看 起 来 它 似乎 很 适合 做 时 钟 。 但 这 个 计数 器 是 跟 CPU 的 ， 并 
且 在 运行 时 可 能 会 受到 节能 或 其 他 因素 的 影响 。 也 就 是 说 ,不同 的 CPU 会 互相 偏离 。 也 不 能 跟 钟 
表 时 间 保 持 一 致 。 

最 后 还 有 高 精度 事件 计时 带 (HPET) 这 种 计时 需 是 最 近 几 年 才 出 现 的 ,有 助 于 人 们 用 较 老 
的 时 钟 便 件 更 好 地 计时 。HPET 使 用 至 少 10MHz 的 计时 器 , 所 以 其 精度 至 少 应 该 是 lus 一 一 但 它 并 
不 是 在 所 有 人 硬件 上 都 可 用 ， 也 不 是 所 有 操作 系统 都 支持 。 

如 果 这 些 内 容 看 起 来 有 点 乱 ， 那 是 因为 它们 本 来 就 乱 。 好 在 Java 平 台 提 供 了 可 以 使 用 它们 的 
工具 一 一 它 把 对 硬件 和 OS 支持 的 依赖 隐藏 到 特定 的 机 器 配置 里 。 然 而 试图 隐藏 依赖 项 的 做 法 并 
没有 完全 成 功 。 


6.4.2 麻烦 的 nanoTime() 


Java 中 有 两 个 获取 时 间 的 方法 : System.currentTimeMillis() 和 Svstem.nanoTime() ， 
后 面 一 个 用 于 测量 比 毫秒 更 精确 的 时 间 。 表 6-1 总 结 了 它们 两 个 的 主要 差异 。 
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表 6-1 Java 内置 时 间 获 取 方 法 的 比较 


currentTimaeMillis'l ) nanoTimel) 
解析 度 为 毫 种 级 纳 种 级 引用 
几乎 所 有 情况 下 都 跟 钟 表 时 间 相 符 可 能 侦 离 钟表 时 间 


如 果 表 6-1 中 对 nanoTime () 的 描述 让 它 看 起 来 有 点 像 计 时 器 , 那 就 对 了 , 因为 如 今 在 大 多 数 
操作 系统 上 ， 它 的 时 间 尘 都 是 CPU 计数 钟 一 一 TSC。 

nanoTime() 的 输出 是 相对 于 某 个 固定 时 间 的 。 也 就 是 说 必须 用 它 记 录 间 隔 期 ， 用 
nanoTime () 的 返回 结果 减 去 之 前 调用 得 到 的 返回 结果 。 下 面 这 段 代 码 来 自 后 面 的 一 个 研究 案 
例 ， 愉 好 表明 了 这 种 情况 

long t0 = System.nanoTime |(); 

doLoopl (); 

londg 上 1 = System.nanoTime (}):; 


1 &@&l = tl1 = 七 日 ; 

el 是 doLoop1 () 执行 所 用 的 时 间 ( 以 纳 秒 为 单位 )。 

要 在 性 能 调 优 中 正确 使 用 这 些 方 法 ,必须 对 nanoTime() 的 行为 有 所 了 解 。 代 码 清单 6-1 输 出 
了 毫秒 计时 器 和 纳 秒 计 时 髓 ( 通常 由 TSC 提 供 ) 之 间 的 最 大 偏离 。 


代码 清单 6-1 时 间 偏 离 


private static void runWithspin{(string[] args) 1 
long nowNanos = 0, startNanos = 0; 
long StartMillis = System.currentTimeMillis'(); 


long nowMillis = startMillis:; 将 startmanos 在 
while (startMillis == nowMillis}) | ge 宫 秒 边界 上 对 齐 


startNanos = System.nanoTime'():; 
nowMillis = System.currentTimeMillist():; 
| 
startMillis = nowM1illis; 
double maxDrift = 0; 
long lastMillis:; 


while (true) | 
lastMillis = nowMillias; 
while (nowMillis - lastMillis < 1000) { 
nowNanog = System. nanoTime(); 
nowMillis = Syastem.currentTimeMilliael(); 
} 
long durationMillis = nowMillis - startMillisa; 
double driftNanos = 1000000 * 
(((double) (nowNanos ~- atartNanos)) / i1000000 = durationMillias):; 
if (Math.abs ldriftNanos) > maxDrift) | 
System.out .println(l"Now = Start = "» durationMillis 
+" driftNManos = "* driftNanos); 
maxDrift = Math.absasldriftNanoa); 
} 
| 
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这 段 代码 会 输出 可 观测 到 的 最 大 偏离 ， 并 且 证 明 其 表现 与 操作 系统 的 相关 度 很 高 。 下 面 是 
Linux 上 的 一 段 输出 : 


1000 driftNManos = 14.99999996212864 


NOow - Start = 

Now = Start = 3000 driftNancs = -B86.99999989403295 

New = Start ss 8000 driftNanocs = -89.0000001163571]1 

Now = Start = S0000 driftNanos = -92.00000204145908 

Now = Start = 67000 driftNanocogs = -=-96.000003395602]1 

Now = Start = 113000 driftNanocs = -98.00000407267362 

NOW - Start = 136000 drifttNanocs = -98.99999713325176 z 

Now - Start = 150000 driftNanos = -101.0000123642385 注意 ariftNanos 
Now - Start = 497000 driftNancs = -2035.000012256205 从 -2035 到 20149 
Now - Start = 1006000 drittNanoae = 20149.99999664724 出 现 了 一 个 非常 
Now - Start = 1219000 driftNanos = 44614.00001309812 大 的 跳跃 


这 里 还 有 一 个 装 在 相同 硬件 上 的 老 Solaris 上 的 输出 结果 : 

1000 driftNanos = 65961.0000000157 

2000 GriftNanocoes = 130928.0000000399 

3000 driftNanos = 197020.9999999497 

000 driftNanos = 261826.99999981196 

5000 driftNanos = 328105.9999999343 : 
E000 driftNanos = 393130.99999981205 间隔 很 平滑 
7000 driftNanos = 458913.9999998224 

a8000 driftNanocgs = S24811 .9999996561 

S5000 driftNanocos = 590093.9999992261 

10000 driftNanocs = 656146 .9999996916 

NOW = Start 11000 driftNancs = 721020.00000086256 

Now = Start 12000 driftNanocs = T786994 .0000000#97 


注意 看 最 大 值 的 增长 ， 在 Solaris 上 很 稳定 ， 而 在 Linux 上 相当 一 段 时 间 内 看 起 来 都 OK， 然 后 
出 现 了 大 的 跳 路 。 我 们 在 选择 示例 代码 时 相当 认真 ， 尽量 避免 创建 额外 的 线程 ， 甚至 对 象 ， 以 将 
平台 的 干预 降 到 最 低 ( 比如 说 ， 没 有 对 象 的 创建 就 意味 着 不 会 做 垃圾 收集 )， 但 即便 如 此 ， 我 们 
还 是 能 看 到 JVM 的 影响 。 

最 终 证 实 Linux 时 序 上 出 现 的 跳 路 是 由 不 同 CPU 上 的 TSC 计 数 器 之 间 的 差异 造成 的 ,JVM 会 定 
期 挂 起 正在 运行 的 Java 线 程 ， 并 将 它 迁 移 到 不 同 核心 上 。 所 以 程序 代码 会 见 到 不 同 CPU 计 数 硕 上 
的 差异 。 

这 就 是 说 对 于 间隔 较 长 的 时 间 ，nanoTime() 基 本 上 是 不 可 信 的 。 只 能 用 它 测 量 较 得 的 时 间 
间隔 ， 较 长 (宏观 ) 的 时 间 间 隔 应 该 用 currentTimeMi11is() 重 新 校准 。 

要 充分 向 提 性 能 调 优 ， 即 要 有 扎实 的 测量 理论 ， 还 需要 知道 实现 细节 。 


6.4.3 时 间 在 性 能 调 优 中 的 作用 


要 做 好 性 能 调 优 ,你 必须 知道 该 如 何 解读 代码 运行 期 间 得 到 的 测量 记录 , 也 就 是 说 你 必须 明 
日 在 Java 平 台 上 得 到 的 时 间 测 量 续 果 的 局 限 性 。 

精确 度 

时 间 的 量 通 常 被 冠 以 与 其 最 接近 的 某 一 刻度 单位 。 这 被 称 为 测量 的 精确 度 。 比 如 说 ， 测 
量 时 间 经 常 精确 到 毫秒 。 如 果 重 复 围绕 同一 数值 进行 测量 ， 其 结果 范围 很 小 ， 则 该 计时 器 是 
精确 的 。 


Now = Start 
Now = Start 
NOW ~ Start 
Now =- Start 
Now = Start 
Now = Start 
NOW = Start 
Now = Start 
NoWw - Start 
NOW = Start 


| | | 


| | 


中 | | | 
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精确 度 是 对 给 定 测量 中 所 包含 的 随机 噪音 量 的 度量 。 我们 假定 对 一 段 代码 的 测量 结果 是 正 态 
分 布 的 。 那 么 精确 度 通 常 是 宽度 为 95% 的 置信 和 区间。 

准确 度 

测量 ( 指 测量 时 间 ) 的 准确 度 是 取得 接近 真实 值 的 测量 结果 的 能 力 。 实 际 上 ， 真实 值 一 般 不 
可 知 ， 所 以 准确 度 可 能 比 精 确 度 更 难 确定 。 

准确 度 是 对 测量 中 的 系统 性 错误 的 度量 。 可 能 存在 准确 但 不 太 精确 的 测量 结果 ( 所 以 基本 读 
ss 


数 是 正确 的 ， 但 有 随机 的 环境 噪音 )。 也 可 陛 征 在 精 


ead : pp pr | 计时 器 测量 的 ， 其 值 为 5945 纳 
秒 ， 那 真实 值 应 该 介 于 3945 一 7945 纳 秒 之 间 (95% 的 可 能 性 )。 当心 那些 看 起 来 过 于 精确 的 数 
值 ， 必 须 随时 对 测量 结果 的 精确 度 和 准确 度 进 行 检查 。 


粒度 

系统 真正 的 粒度 是 最 快 计 时 器 的 频率 一 一 很 可 能 是 10 纳 秒 范围 内 的 中 断 计 时 器 。 这 有 时 被 称 
为 可 办 别 能 力 ， 可 以 肯定 “几乎 一 起 发 生 ， 但 时 间 不 同 ” 是 两 个 事件 最 短 的 发 生 间 隔 。 

在 我 们 路 过 操作 系统 、 虚 拟 机 和 类 库 代 码 的 不 同 层 面 时 , 已 经 不 太 可 能 辨别 这 些 极 短 的 时 间 
间隔 。 在 大 多 数 情况 下 ， 应 用 程序 开发 人 员 是 得 不 到 这 些 特 别 短 的 时 间 间 隔 的 。 

分 布 式 网 络 计 时 

我 们 对 性 能 调 优 的 大 部 分 讨论 都 是 以 单机 上 的 系统 为 中 心 的 。 但 你 应 该 知道 ， 当 涉及 网 络 上 
的 系统 调 优 时 , 会 有 一 些 特别 的 问题 网 络 上 的 同步 和 计时 并 不 容易 , 而 且 不 仅仅 是 在 互联 网 上 ， 
即便 是 以 太 网 也 会 出 现 这 些 问题 。 

详细 讲解 分 布 式 网 络 计时 超出 了 本 书 的 范围 , 但 你 应 该 知道 , 通常 来 说 , 很 难得 到 用 于 跨越 几 
台 机 器 的 工作 流 的 准确 时 序 。 另 外 ， 即便 NTP 这 样 的 标准 协议 对 于 高 精度 工作 来 说 准确 度 也 不 够 。 

在 开始 讨论 垃圾 收集 之 前 ， 我 们 先 看 一 个 前 面 提 到 过 的 例子 一 一 缓存 对 代码 性 能 的 影响 。 


6.4.4 ”案例 研究 ; 理解 缓存 未 命中 


对 于 很 多 吞吐 量 较 高 的 代码 来 说 ， 影 响 性 能 的 一 个 主要 因素 就 是 一 级 缓存 未 命中 的 数量 。 

代码 清单 6-2 中 的 代码 操作 1MB 的 数组 ， 并 输出 执行 两 个 循环 中 之 一 所 用 的 时 间 。 在 第 一 个 
循环 中 ， 每 隔 16 个 条 目 对 int 数 组 中 的 元 素 加 1。 一 级 缓存 的 一 个 缓存 行 中 通常 有 64 个 字 节 ( 在 
32 位 JVM 上 ，Java 的 int 是 4 个 字 节 )， 所 以 这 意味 着 每 次 会 读 取 一 个 缓存 行 ( 64=16*4 )。 


public class CacheTeater 1 


private final int ARR SIZE = 1 * 1024 * 1024; 
private final int[] arr = new int [ARR SIZE]; 
, 由 理 每 个 条 
private void doLoop2() 1 每 个 条 目 
for lint i1=0; i<arr.length; i++) arrlil])++; 
} 
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private void doLocop1() I ] 处 理 每 个 缓存 行 
i=s0; i<zarr.length; i 4 16) arr[i]++; 二 


for (int i=0; 


} 


private void run() | 


for (int i=0; i<10000; i++) 1 和 
doLoopl () ; == 代码 热身 


doLoop2()| 

| 

for (int i=0; i<100; i++*) 1{ 
long t0 = Syatem.nanoTime (); 
doLoopl 1(); 
long tl = System.nanoTime (); 
doLoop2 ()， 
long t2 = Syastem.nanoTime (); 
long el = tl1 - tO; 
long el2 二 t2 = tl1} 
System.out .println("Loopl: "+ el + nanos ; Loop2: "+ el2); 

} 

| 


public static void mainlstring[] args) | 
CacheTester ct = new CacheTester'():; 
ct .runt(): 
} 
| 


注意 ， 在 你 得 到 准确 结果 之 前 应 该 让 代码 热 热 喘 ， 以 便 让 JVM 对 你 感 兴 趣 的 方法 进行 编译 。 
我 们 会 在 6.6 节 讨论 更 多 与 代码 热 映 相关 的 内 容 。 

第 二 个 循环 ，doLoop2 () 给 数组 中 的 每 个 元 素 加 1， 所 以 看 起 来 它 做 的 工作 是 doLoop1 () 的 
16 倍 。 下 面 是 在 笔记 本 上 运行 这 段 代码 得 到 的 结果 ， 

Loopl: 634000 mancs ;} Loop2: 6868000 

Loopl: 801000 nanos ; Loop2: 952000 

Loopl: 676000 nanos ; Loopz2: 330000 

Loopl: 762000 nanos ; Loop2: 869000 

Loopl: T706000 nanocs ;1 Loop2:; 798000 


1 hull pe I Ee 加 be 了 上 i 1] nm 站 也 = niiom.. pr =" 9 -~ A - 加 站 ee 
时 = i oe- i ee a | ph ra E 有 es . ' se 下 a 加 上 mil a | y - Ge 
| 1 a eal rh . 码 的 pe 证 二 | 和 -性 1 有 人 | | be lL 和 i li 让 生生 ee ! ns 成 | ,| EF b | Hl 
Dl 日 = 去 | = in 让 人 i | tl 
和 i . | i 
项 1 


油漆 囊 本 所 有 需 履 值 都 很 天 齐 ， MR 
nanoTime() 最 终 所 调用 的 ) 仅仅 返回 了 一 个 微 秒 整 教 值 ” 一 机 秒 是 1000 纳 秒 。 因 为 这 个 结 
oon 所 以 我 们 猜测 在 OS 义 的 底层 系统 调用 只 有 徽 秒 级 的 精度 , 实际 上 ， 


从 这 个 结果 来 看 ，doLoop2 () 所 用 的 时 长 不 是 aoLoopl() 的 16 倍 。 这 表明 内 存 访问 在 总 体 
性 能 配置 中 占有 支配 性 地 位 。doLoop1 1() 和 doLoop21) 读 取 绥 存 行 的 次 数 相同 ， 而 修改 数据 所 
用 的 CPU 周期 只 占 整 体 时 间 的 一 小 部 分 。 

我 们 先 来 回顾 下 Java 时 间 系 统 的 要 点 。 


时 6 
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口 大 多 数 系统 内 部 都 有 几 个 不 同 的 时 钟 。 

口 毫秒 计时 器 是 安全 可 菲 的 。 

口 更 高 精度 的 时 间 需 要 仔细 处 理 以 防止 出 现 侦 高 。 

口 你 需要 知道 计时 测量 的 精确 度 和 准确 度 。 

我 们 下 一 个 将 要 讨论 的 是 Java 平 台 的 垃圾 收集 子 系统 。 这 是 性 能 的 决定 性 因素 中 非常 重要 的 一 
部 分 ， 并 且 它 有 很 多 可 调节 的 部 分 ， 对 于 做 性 能 分 析 的 开发 人 员 来 说 都 可 以 成 为 非常 重要 的 工具 。 


6.5 垃圾 收集 


内 存 自动 管理 是 Java 平 台 最 重要 的 组 成 部 分 之 一 。 在 出 现 Java 和 .NET 这 样 的 托管 平台 之 前 ， 
开发 人 员 把 大 部 分 时 间 都 用 在 追踪 不 完善 的 内 存 处 理 引 发 的 bug 上 了 了。 

然而 近年 来 ， 内 存 自 动 分 配 技 术 发 展 的 如 此 先进 可 徘 , 已 经 变 得 让 人 无 法 察觉 了 ， 因 此 大 部 
分 Java 开 发 人 员 不 知道 Java 平 台 的 内 存 管理 是 如 何 完成 的 ， 不 知道 可 以 使 用 哪些 选项 ， 也 不 知道 
如 何在 框架 限定 内 进行 优化 。 

这 说 明 Java 的 做 法 取得 了 成 功 , 大 多 数 开发 者 不 知道 内 存 和 GC 系统 的 细节 是 因为 他 们 没 必要 
知道 。 虚拟 机 在 这 方面 做 得 非常 棒 ， 在 处 理 大 多 数 应 用 时 都 不 用 特别 调整 ， 所 有 大 多 数 应 用 从 设 

本 节 我 们 将 讨论 在 确实 需要 做 些 调 整 的 情况 下 你 能 做 什么 。 我 们 会 给 出 基本 原理 , 解释 为 了 
运行 Java 进 程 该 如 何 处 理 内 存 ， 并 探索 标记 和 清除 集合 的 基础 ， 再 讨论 两 个 工具 一 一 jmap 和 
VisualVM。 最 后 介绍 两 个 收集 器 一 一 并 发 标记 清除 ( Concurrent Mark-Sweep， 人 简称 CMS ) 和 新 
的 垃圾 优先 (Garbage First， 简 称 G1 ) 收集 器 。 

也 许 你 有 个 服务 器 端 程 序 耗 光 了 内 存 ， 或 者 承受 着 长 时 间 中 断 的 痛 苗 。 在 6.5.3 节 讨论 jmap 
时 ,我 们 将 会 告诉 你 一 个 查看 类 是 否 占 用 大 量 内 存 的 简单 办 法 。 我 们 还 会 教 你 使 用 控制 虚拟 机 内 
存 配置 的 选项 开关 。 

先 从 基本 算法 开始 吧 。 


6.5.1 基本 算法 


标准 的 Java 进 程 既 有 栈 又 有 堆 。 栈 保存 原始 型 局 部 变量 ( 引用 型 局 部 变量 会 指向 以 堆 方式 分 
配 的 内 存 )。 推 保存 要 创建 的 对 象 。 图 6-4 展 示 了 各 种 类 型 变量 存储 的 位 置 。 


图 6-4 ” 堆 和 栈 中 的 变量 


1 二 
OEEEE 


1S0 第 6 章 理解 性 能 调 优 


注意 ,对象 的 原始 型 域 仍然 分 配 在 堆 内 的 地 址 上 。Java 平 台 对 堆 内 存 回收 和 再 利用 的 基本 算 
法 被 称 为 标记 和 清除 ， 应 用 程序 中 代码 已 经 不 再 使 用 它 了 。 


6.5.2 ”标记 和 清除 


标记 和 清除 是 最 简单 、 也 是 出 现 最 早 的 垃圾 收集 算法 。 业 内 还 有 其 他 内 存 上 自动 管理 技术 ， 比 
如 Perl 和 PHP 等 语言 采用 的 引用 计数 ”， 有 人 说 它 更 简单 ， 但 它 是 不 需 做 垃圾 收集 的 方案 。 

最 简单 的 标记 和 清除 算法 会 暂停 所 有 正在 运行 的 线程 ， 并 从 一 组 “ 活 ” 对 象 一 一 在 任何 用 户 
线程 的 任何 堆栈 帧 中 存在 引用 (不管 是 局 部 变量 、 方 法 参数 、 临 时 变量 ， 还 是 某 些 非 常 少 见 的 情 
况 ) 的 对 象 一 一 开始 明 历 其 引用 树 ， 标 记 出 过 历 路 径 上 的 所 有 活 对 象 。 过 历 完成 后 ,所 有 没 被 标 
记 的 都 被 当做 垃圾 ， 可 以 回收 ( 清除 )。 注意， 被 清除 的 内 存 不 会 还 给 操作 系统 ， 而 是 还 给 JVM。 

Java 平 台 对 基本 的 标记 清除 方法 进行 了 改进 ,采用 “分 代 式 垃圾 收集 "。 在 这 种 方法 中 ， 会 
根据 Java 对 象 的 生命 周期 将 堆 内 存 划 分 为 不 同 的 区 域 。 在 对 象 的 生存 期 内 ， 对 它 的 引用 可 能 指向 
内 存 中 几 个 不 同 区 域 ( 如 图 6-5 所 示 )。 在 垃圾 收集 过 程 中 ， 可 能 会 将 对 象 移动 到 不 同 区 域 。 


图 6-5 ”内 存 中 的 伊人 向 园 、 幸 存 者 乐园 、 终 身 颐 养 园 和 PermGen 区 
这 样 做 是 因为 根据 对 系统 运行 时 期 的 研究 ， 发 现 对 象 的 生存 期 或 者 较 短 ， 或 者 很 长 。Java 平 
台 把 堆 内 存 划 分 为 不 同 区 域 可 以 充分 利用 对 象 生命 周期 的 这 种 特点 。 
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Java 和 .NET 经 常 受到 这 样 的 批评 : 标记 和 清除 式 的 垃圾 收集 不 可 避免 地 会 导致 世界 停 转 
( 所 有 用 户 线程 都 必须 停止 )， 而 且 这 种 暂停 的 时 长 是 不 确定 的 。 

其 实 这 个 问题 被 压 大 了 。 对 于 服务 器 端 软件 来 说 ,应 用 程序 不 会 在 意 垃 圾 收集 引起 的 暂停 。 
为 了 避免 暂停 或 完全 收集 而 精心 制作 解决 方案 完全 是 赁 空想 象 -一 -除非 经 过 认真 分 析 , 发 现 全 
内 存 收集 时 间 真 的 存在 问题 ， 才 应 该 避免 ， 


中 引用 计数 就 是 为 每 个 内 存 对 象 维护 一 个 引用 数值 ， 当 有 新 的 引用 指向 该 对 象 时 则 和 将 其 引用 计数 加 一 ， 销 毁 时 则 减 
一 。 当 引用 计数 为 零 时 就 收回 该 对 章 占 用 的 内 存 资 源 。 这 种 方式 虽然 简单 但 存在 两 个 问题 ， 每 次 内 存 对 象 被 引 
用 或 引用 被 销毁 时 必须 修改 引用 计数 ， 造 成 整体 性 能 消耗 ;出现 循环 引用 时 难以 处 理 。 一 一 详 者 注 
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1. 内 存 区 域 
JVM 为 存储 不 同 生命 周期 阶段 的 对 象 将 内 存 分 成 了 几 个 不 同 区 域 。 
口 伊 名 园 一 伊甸园 是 对 象 最 初 降生 的 堆 区 域 ， 并 且 对 大 多 数 对 象 来 说 ， 这 里 是 它们 唯一 
存在 过 的 区 域 。 
口 幸存 者 乐园 一 一 这 里 通常 有 两 个 空间 (或 者 也 可 以 认为 是 被 分 成 两 半 的 一 个 空间 )。 从 伊 
亿 园 幸存 下 来 的 对 象 会 被 挪 到 这 里 。 它 们 有 时 候 被 称 为 从 何 而 来 和 到 哪里 去 。 除 非 正 在 
执行 垃圾 收集 ， 和 否则 总 有 一 个 幸存 者 室 间 是 空 的 ， 原 因 会 在 后 面 给 出 。 
终身 颐养 园 一 一 终身 制 空 间 ( 即 老 一 代 ) 是 那些 “足够 老 ” 的 幸存 对 象 的 归宿 ( 从 幸存 
者 空间 挪 过 来 的 )。 在 年 轻 代 收 集 过 程 中 是 不 会 碰 终 身 制 内 存 的 。 
D PermGen 一 一 这 是 为 内 部 结构 分 配 的 内 存 ， 比 如 类 定义 。PermGen 不 是 严格 的 堆 内 存 ， 并 
且 普 通 的 对 象 最 后 不 会 在 这 里 结束 。 
就 像 前 面 提 到 的 , 这 些 内 存 区 域 的 垃圾 收集 方式 也 不 尽 相 同 。 具 体 来 说 有 两 种 方式 : 年 轻 代 
收集 和 完全 收集 。 
2, 年 轻 代 收集 
年 轻 代 收 集 只 会 清理 “年 轻 的 ”空间 (伊甸园 和 幸存 者 乐园 )。 其 过 程 相当 简单 。 
口 在 标记 阶段 发 现 的 所 有 仍然 存活 的 年 轻 对 象 都 会 被 挪 走 : 
@ 那些 足够 老 的 对 象 ( 从 次 数 足够 多 的 GC 中 幸存 下 来 的 ) 进入 终身 颐养 园 ; 
田 剩 下 那些 年 轻 的 存活 对 象 进 入 幸存 者 乐园 里 空 着 的 空间 。 
口 最 后 ， 伊 何 园 和 最 近 膳 空 的 幸存 者 乐园 就 可 以 重用 了 ， 因 为 它们 里 面 已 经 全 是 垃圾 了 。 
当 伊 个 园 满 了 的 时 候 就 会 触发 一 次 年 轻 代 收集 。 注 意 ， 标 记 阶 段 必须 遍历 整个 生存 对 象 图 。 
也 就 是 说 如 果 有 个 年 轻 对 象 被 一 个 终身 对 象 引 用 了 , 终 导 对象 所 持 有 的 引用 也 必须 被 扫描 到 并 标 
记 上 。 理 则 只 被 终身 对 象 引 用 的 伊甸园 对 象 可 能 会 出 问题 。 如 果 标 记 阶 段 不 是 全 遍历 ， 这 个 伊 甸 
园 对 象 就 再 也 看 不 到 了 ， 而 且 不 可 能 对 它 做 出 正确 处 理 。 
3. 完全 收集 
当年 轻 代 收集 不 能 把 对 象 放 进 终身 颐养 园 时 ( 空间 不 够 了 )， 就 会 触发 一 次 完全 收集 。 根 据 
老年 代 所 用 的 收集 器 , 这 可 能 会 牵涉 到 老年 代 对 象 的 内 部 迁移 。 这 样 做 是 为 了 确保 必要 时 能 从 老 
年 代 对 象 所 占 的 内 存 中 给 大 的 对 象 腾 出 足够 的 空间 。 这 被 称 为 压缩 。 
4. 安全 点 
要 想 做 垃圾 收集 ， 至 少 得 让 所 有 应 用 线程 暂停 一 会 儿 。 但 是 线程 不 可 能 为 了 GC 说 停 就 停 。 
所 以 它们 给 执行 GC 留 出 了 特定 的 位 置 一 一 安全 点 。 常 见 的 安全 点 是 方法 被 调用 的 地 方 (“调用 
点 "”)， 不 过 也 有 其 他 安全 点 。 为 了 执行 垃圾 收集 ， 所 有 应 用 程序 线程 都 必须 停 在 安全 点 上 。 
我 们 暂停 一 下 ， 先 介绍 一 个 简单 的 工具 一 一 jmap， 它 能 帮 你 和 弄 清 楚 程序 运行 时 的 内 存 使 用 
情况 ， 以 及 所 有 内 存 都 用 在 哪里 。 我 们 后 续 还 会 介绍 一 个 更 先进 的 GUI 工 具 ,， 但 既然 很 多 问题 都 
可 以 用 非常 简单 的 命令 解决 ， 你 最 好 应 该 知道 如 何 使 用 ， 而 不 是 直接 就 使 用 GUI 工 具 。 
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6.5.3 jmap 


Oracle JVM 自 带 了 一 些 简单 的 工具 ， 可 以 帮 你 了 解 运 行 中 的 进程 。jmap 是 其 中 最 简单 的 一 
个 , 用 来 显示 Java 进 程 的 内 存 映 射 ( 它 也 能 分 析 Java 核 心 文件 ， 其 至 能 连 到 远程 调试 服务 器 上 )。 
让 我 们 回 到 电子 商务 服务 右 端 应 用 程序 的 例子 上 ， 用 jmap 对 它 进 行 一 番 探 索 。 

1. 默认 视图 

jmap 最 简单 的 用 法 是 查看 连接 到 进程 里 的 本 地 类 库 。 除 非 你 的 应 用 程序 里 有 很 多 JNI 代 码 ， 
否则 这 种 用 法 通常 没什么 用 ,但 我 们 还 是 会 演示 一 下 ， 以 兔 你 忘 了 指定 jmap 选 项 时 被 它 搞 糊 涂 : 

SS jmap 19306 

Attaching to process ID 19306, please wait... 

Debugger attached successtftully. 

Server compiler detected. 

JYM verBion is 20.0-b11 


0x08048000 A6HK /uar/local/jJava/sunjdk/l1.6.0 25/bin/java 
Ox55555000 108K /ilib/ld-2.3.4.80 
,Some entries omitted 
OxS563eB000 S35 /liib/libnss db.s0.2.0.0 
OxTedl8000 9aK /usr/local/jJava/sunjdk/1.6.0 25/jre/lib/i386/l1ibnet .so 


Oxa80cfE3000 2102K /usr/local/kerberog/mitkrbs/l1 .4 .4/1ib/ 
libgss all.so.3.1 
vxs80dcf000 工 间 寻 三 玖 :usr/local/kerberos/mitkrb5/1.4.4/1ib/libkrbs.so.3.2 


一 般 用 得 比较 多 的 是 -heap 和 -histo 选 项 ， 下 面 我 们 就 来 讨论 这 两 个 选项 。 

2. 堆 视图 

使 用 -heap 选 项 时 ，jmap 会 抓 取 进程 当前 的 堆 快 照 。 在 输出 结果 中 能 看 到 构成 Java 进 程 堆 内 
存 的 基本 参数 。 

堆 的 大 小 是 年 轻 代 、 老年 代 加 上 PermGen 区 的 总 和 。 但 在 年 轻 代 内 部 有 伊甸园 和 幸存 者 乐园 ， 
并 且 我 们 还 没 告诉 你 这 些 区 域 的 大 小 之 间 有 什么 关系 。 这些 区 域 的 相对 大 小 是 由 一 个 叫做 幸存 比 
例 的 数值 决定 的 。 

我 们 来 看 一 些 和 输出 样 例 。 你 能 在 其 中 看 到 伊甸园 、 幸 存 者 乐园 ( 标签 为 From 和 mo )、 终 身 喇 
养 网 (01q4 Generation ) 以 及 一 些 相关 信息 

3 jmap -heap 22186 

Attaching to process ID 22186, please Wait... 

Debugger attached successtully. 


Server compiler detected. 
JVM version ig 20.0-b11 


using thread-local object allocat Lon. 
Parallel GC with 13 thread (as) 


中 Java 核 心 文件 ( Java core file ) 主要 保存 各 应 用 线程 在 某 一 时 刻 的 运行 位 置 ， 即 JVM 执 行 到 哪个 类 、 哪 个 方法 及 哪 
一 行 上 。 它 是 一 个 文本 文件 ,打开 后 可 以 看 到 每 一 个 线程 的 执行 栈 ， 以 及 stack trace 的 显示 。 一 般 Java 程 序 遇 到 致 
荫 问 题 ， 在 JVM 死 掉 之 前 会 产生 两 个 文件 ， 其 中 就 有 Java 核 心 文件 ， 另 一 个 是 HeapDump 文 件 。 有 时 为 了 调试 或 
查找 性 能 问题 也 会 手工 生成 这 两 个 文件 。 一 一 译 者 广 
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Heap Configuration: 


MinHeapFreeRatio = 40 

MaxHeapFreeRatio = 70 

MaxHeapSize = S36870912 (512.0MB) 

NewSize = 1048576 (1 .0MB) 

MaxNewSize 三 4294901760 (4095.937SMB) 

Oldsize = 上 LI94304 (4 .0MB) 

NewRat1ioc = 2 | 伊甸园 = (From+To) * 
SurvivorRatio = 本 一 幸存 比例 

PermSize = 16777216 (16.0MB) 


MaxPermSize = 67108864 (64 .0MB) 
Heap Usage: 


PS Young Generation 
Eden Space: 
capacity = 163774464 (156 .1875MB) 


uBed = S652576 (S55.935455322265625MB) 
free = 105121888 (100.25204467773438MB) 


35.81301661289S516$ Used 

From SpAace: 
capacity T012352 (6.6875MB) 
USed al44688 (3.9526824951171875MB) 
free = 2867664 (2.7348175048828125MB) 
59.10553263726636 Used 

To Space: 
Capacity = T274496 (6.9375MB) 


伊甸园 = (From+To) " 
幸存 比例 


free = T274496 [6.9375MB) 


0.0$% used ec 一 
PS Old Generation | To 空间 当前 为 空 


capacity = 89522176 (85.37S5MB) 


刀 瑟 忆 避 = 158272 (5.87298583984375MB) 
free = B3363904 (79.50201416015625MB) 


.07904637170571$% wsed 

PS Perm Generation 
capacity = 30146560 (28 .7SMB) 
UsSed 30086280 128.69251251220703MB) 
free 80280 (0.05748748779296875MB) 
99.80004352072011%®% used 


尽管 空间 的 基本 构成 可 能 会 非常 有 用 , 但 在 这 副 图 里 看 不 到 堆 里 面 有 什么 。 如 果 能 看 到 是 哪 


些 对 象 占用 了 内 存 中 的 空间 ， 你 就 知道 内 存 都 到 哪里 去 了 。jmap 恰 好 提供 了 一 个 柱状 图 模式 ， 
可 以 让 你 看 到 这 些 数据 的 简单 统计 结果 。 


3. 柱状 视图 
柱状 视图 显示 了 系统 中 每 个 类 型 的 实例 (还 有 一 些 内 部 实体 ) 占用 的 内 存量 。 各 个 类 型 按 使 


用 内 存 多 少 排列 ， 这 样 就 比较 容易 看 到 最 大 的 内 存 猪 。 


当然 ， 如 来 所 有 内 存 祁 交 给 了 了 框 商 和 平台 类 , 这 里 可 能 就 没 你 什么 事 了 。 但 如 果真 有 一 个 你 


的 类 ， 有 了 这 些 信息 便 能 更 好 地 干预 它 的 内 存 占用 。 


小 小 的 警告 ，jmap 使 用 类 型 内 部 名 称 。 比 如 字符 数组 会 写成 [Cc， 类 对 人 象 的 数组 会 显示 。 


只 自 在 读书 @ 
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$$ jmap -histo 22186 | head -30 


num #instances #bvtes class name 

1 和 271719 311l12412 [©C 

Fr T6877 1a924304 [B 

3 20817 12188728 [Liava. lang.Object:; 

as 2520 10547976 com.company.cache.Cache$AccountInfo 

S53 439499 91455€60 Java.lang.Sstring 

本 : 本 总 二 后 五 Tol19800 [ 工 

了 64466 S677912 <constMethodKlass> 

8 96840 4333424 <methodKlass> 

3 56990 3384504 <symbolKlasass 

10: 6990 2944272 <conastant PoolKlase> 

11: 上 991 1855272 <instanceKlassKlasss> 

lz: 2598 人 0 1l247040 <conastantPoolCacheklasss 

13; 17250 1209984 java.nio.HeapCharBuffer 

14: 13515 1173568 [Ljava.util.HashMap$Entry:; | VM 内 部 对 象 
15: 号 733 了 了 864D Java.lang.reflect .Method 和 类 型 信息 
1€: 17842 713680 Java.nio.HeapByteBurffer 

17: 7433 713568 java.lang.Class 

18: 10771 678664 【号 

19: 1543 A89368 <methodDataKlass> 坷 - 
20: 10620 十 与 百 二 3 石 [ {I 

21: 18285 38840 Java.util .HashMap$Entry 

必 : 9985 399400 java.util .HashMap 

23: 13725 329400 Java.util.Hashtable$Entry 

24 : 9839 314848 java.util .LinkedHashMap$Entry 

与: 9793 249272 [Ljava.lang.Sstring:; 

必 扣 : ll19271 241192 [Liava.lang.Class; 

27; 6903 2208936 java.lang.ref .SoftReference 


因为 在 柱状 图 模式 下 输出 的 数据 很 多 ， 所 以 上 面 只 显示 了 输出 内 容 的 一 部 分 。 你 可 能 要 用 
grep 或 其 他 工具 来 查看 柱状 图 视图 ， 找 到 感 兴趣 的 细节 。 

输出 中 有 很 多 占用 内 存 的 [c 实 体 。 字 符 数 组 数据 经 常 出 现在 String 对 象 里 (字符 串 的 内 容 
就 存在 那里 )， 所 以 这 不 奇怪 一 一 大 多 数 Java 程 序 里 都 有 很 多 字符 串 。 但 从 柱状 图 中 还 能 看 出 其 
他 有 趣 的 事情 。 先 来 看 看 下 面 两 个 。 

前 几 个 实体 里 唯一 一 个 应 用 类 是 cache$AccountInfo 一 一 其 他 全 是 平台 或 框架 类 型 一 
所 以 它们 是 开发 人 员 可 以 完整 控制 的 最 重要 的 类 型 。AccountInfo 对 象 占 了 很 多 空间 一 一 大 概 
2 500 个 实体 占 了 10.5 MB (或 者 每 个 账号 占 4 KB )。 对 于 账号 细节 来 说 这 实在 是 很 多 。 

这 个 信息 真 的 非常 有 用 。 你 已 经 知道 代码 里 什么 占 内 存 最 多 了 。 假 如 老板 现在 过 来 告诉 你 ， 
因为 大 规模 促销 ， 一 个 月 内 系统 客户 数 可 能 要 暴 增 10 倍 。 你 知道 这 可 能 会 给 系统 增加 很 多 压 
力 accountInfo 对 和 可 是 个 凶猛 的 大 家 伙 。 虽 然 你 有 点 担心 ， 但 至 少 你 已 经 开始 分 析 这 个 
问题 了 。 

jmap 输 出 的 信息 可 以 作为 潜在 问题 处 理 决策 流程 的 辅助 输入 。 你 是 不 是 应 该 把 账号 缓存 分 
开 , 减少 该 类 型 保存 的 信息 项 ,或 者 买 更 多 的 内 存 给 服务 器 装 上 。 在 做 出 任何 决定 之 前 ， 你 还 要 
做 很 多 分 析 工 作 ， 但 已 经 有 个 起 点 了 。 
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柱状 图 模式 下 还 能 看 到 其 他 有 意思 的 事情 ， 这 次 指定 -histo:1live 选 项 。 这 是 告诉 jmap 只 
处 理 存 活 对 象 ， 而 不 是 整个 堆 ( jmap 默 认 会 处 理 所 有 对 象 ， 也 包括 还 没 被 收集 的 垃圾 )。 让 我 们 
看 看 这 次 输出 什么 ， 


$ jmap -hiato:live 22186 | head -7 


num #instancea A class name 
汪 - 2520 10547376 com.company .cache.Cache$AccountInfo 
三 32196 a4919800 [I 
3 5392 4237628 [Ljava.lang.oObject; 
入: 1 对 1 六 台 1 2187368 [CC 


` 广 意 输 出 的 变化 一 一 字符 数据 已 经 从 31MB 降 到 了 2MB 左 右 了 ,证 明 你 第 一 次 看 到 的 String 
对 象 里 有 将 近 三 分 之 二 都 是 等 竺 回收 的 垃圾 。 然 而 账号 对 象 全 是 活 的 ,进一步 证 明了 它们 是 消耗 
内 存 的 主要 力量 。 

使 用 jmap 时 应 该 稍微 道 慎 点 。 进 行 该 操作 时 JVM 还 在 运行 ( 如 果 你 不 走运 ， 还 有 可 能 在 读 
取 快 照 期 间 做 了 垃圾 回收 )， 所 以 你 应 该 多 运行 几 次 , 特别 是 在 你 看 到 任何 奇怪 或 太 好 的 结果 时 。 

产生 离线 导出 文件 

jmap 能 创建 导出 文件 ， 像 这 样 : 

jmap -dump:live,format=b,file=heap.hprof 19306 

导出 结果 可 以 用 来 做 离线 分 析 , 可 以 留 给 jmap 以 后 自己 用 , 也 可 以 留 给 Oracle 的 jhat ( Java 
堆 分 析 工 具 ) 做 高 级 分 析 。 可 异 我 们 没 办 法 在 这 里 全 面 讨论 。 

使 用 jmap 可 以 看 到 一 些 基本 设置 和 程序 的 内 存 占用 。 然 而 要 做 性 能 调 优 ， 一 般 需要 对 GC 子 
系统 有 更 多 控制 ， 其 标准 方式 是 通过 命令 行 参数 ,我 们 来 看 一 些 控制 JVM 的 参数 ,用 它们 使 JVM 
的 行为 更 适用 于 你 的 应 用 程序 。 


6.5.4 与 GC 相关 的 JVM 参数 


JVM 的 参数 非常 多 ( 最少 上 百 个 )， 用 来 定制 JVM 运 行 时 的 行为 。 本 节 我 们 会 讨论 一 些 跟 垃 
圾 收集 有 关 的 选项 ， 后 续 章 广 中 还 会 讨论 其 他 选项 。 


a 二 -3 i i i jE | Tt 全 三 必 二 :二 府 S 时 | 打 沁 
gt 本 | | ds = 一 页 | : 


i jl a hh 记 | 划 让 od 4 

以 -xX: 开头 的 选项 不 是 标准 选项 ， 在 其 JVM 可 能 不 可 用 。 | 

以 -XX: 开 头 的 是 扩展 选项 ， 不 要 随便 使 用 。 供 多 与 性 能 相关 的 寺 项 孝 是 扩展 和 项、 

有 些 选项 相当 于 布尔 型 的 参数 ， 并 且 前 面 有 + 或 -作为 它 的 开关 。 还 有 币 参 数 的 选项 ,此 
如 -XX:CompileThreshold=1000 ( 这 个 方法 会 在 调用 次 数 达 到 1000 之 后 才 被 和 J Th 编译 )。 还 
有 一 些 参 数 ( 包括 很 多 标准 参数 ) 既 没 有 开关 也 不 能 带 参 数 。 en 


表 6-2 中 是 基本 的 GC 选项 ， 还 有 这 些 选 项 的 默认 值 ( 如 果 存 在 )。 
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表 6-2 基本 垃圾 收集 选项 


选 ”项 效果 


_xms< LMB>m 堆 的 初始 大 小 (默认 2 MB) 
-Xmxe< JLMB>m 堆 的 最 大 大 小 (默认 64 MB) 
-Rnmmn< (LMB>m 


堆 中 年 轻 代 的 大 小 


-XxX: -DisablegxplicitGc 让 调用 Syatem.gc (不 产生 任何 作用 


一 个 常用 的 小 技巧 是 把 -xms 和 -xmx 的 大 小 设 成 一 样 的 。 这 样 进程 就 会 用 恰当 的 堆 尺 寸 运 行 ， 
没 必要 在 执行 过 程 中 调整 大 小 ( 可 能 会 引发 意 想不到 的 降 速 )。 


表 中 最 后 一 个 选项 输出 GC 的 标准 信息 到 日 志 中 ， 我们 在 下 一 节 会 讨论 如 何 解 释 这 些 信息 。 
6.5.5 ” 读 懂 GC 日 志 


为 了 充分 利用 垃圾 收集 ， 你 需要 经 常 看 看 子 系统 在 做 什么 。 除 了 基本 的 verbose:gc 标 记 ， 
还 有 很 多 可 以 控制 输出 信息 的 选项 。 

别 拿 GC 日 志 不 当 回 事 儿 ， 你 可 能 时 不 时 地 就 会 发 现 自己 被 输出 信息 淹没 了 。 下 一 节 讨 论 

VisualVM 时 你 会 发 现 ， 有 一 个 可 视 化 工具 可 以 帮 你 看 到 VM 的 行为 ， 这 个 工具 非常 有 用 。 不 管 怎 


样 , 会 读 日 志 以 及 了 解 影响 GC 的 基本 选项 非常 重要 ,因为 有 时 候 你 可 能 没 法 用 GUI 工 具 。 最 常用 
的 GC 日 志 选 项 如 表 6-3 所 示 。 


表 6-3 ”用 于 扩展 日 志 的 额外 选项 
选 项 | 


-XR:+PrintGcCcDetails 


”效果 
关于 GC 更 详细 的 细节 


GC 操 作 的 时 间 惟 
-XX:+PrintechApplicationConcurrentTime 任 应 用 线程 仍然 返 行 的 情况 下 用 在 GC 上 的 时 间 


-XX:+PrintGCDateStamps 


这 些 选 项 组 合 在 一 起 时 ， 会 产生 下 面 这 种 日 志 : 


6.580: [GC [PSYouNngGen: 486784K- >»>7667K(499648K)] 
l292752K-=813636K 1400768K}) ， 0.0244970 secal] 


i 3 : me 
我 们 把 它 分 解 ， 看 看 每 一 部 分 是 什么 意思 : 
<times: [GC [<collector names: <Occupancy at starts> 

-> OCCUpPANCY at end> (<total Slize>))] <tull heap occupancy at starts 
-» <full heap Occupancy at ends (<total heap sizes), 


ai <bause times> Secs 


第 一 块 是 GC 的 发 生 时 间 ， 从 JVM 启 动 开 始 算 ， 到 发 生 时 的 秒 数 。 然 后 是 用 来 收集 年 轻 代 的 
收集 器 名 称 ( PSYoungGen )。 接 着 是 年 轻 代 收 集 前 后 占用 的 内 存 ， 以 及 年 轻 代 的 总 大 小 。 接 着 
是 反映 完全 收集 情况 的 相同 部 分 。 

除了 GC 日 志 选 项 ， 还 有 一 个 选项 如 果 不 经 解释 可 能 会 引起 误解 。 用 选项 -xx:+PrintGc- 
ApplicationstoppedTime 产 生 的 日 志 是 这 样 的 ， 


1 入 
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Application time: 0.9279047 seconds 
Total time for Which application threads were stopped: 0.0007529 seconds 
Application time; 0.0085059 Beconaa 
Total time for which application threads were stopped: 0.0002074 seconds 
Application time: 0.0021318 Seconds 


这 些 并 不 一 定 指 GC 用 了 多 长 时 间 ， 而 是 指 在 一 个 从 安全 点 开始 的 操作 中 ， 线 程 停 了 多 长 时 
间 。 这 包括 GC 操作 ,但 也 包括 其 他 安全 点 操作 ( 比如 Java 6 中 的 偏向 锁 操 作 ”)， 所 以 没有 十 足 把 
握 说 这 是 指 GC 时 长 。 

所 有 这 些 信息 对 记录 日 志和 事后 分 析 都 有 用 , 但 不 容易 做 可 视 化 处 理 。 而 很 多 开发 人 员 在 做 
初始 分 析 时 都 喜欢 使 用 GUI 工具 。 好 在 HotSpot VM ( 标准 的 Oracle VM， 稍 后 讨论 ) 自 带 了 一 个 
非常 实用 的 工具 。 


6.5.6 用 VisualVM 查看 内 存 使 用 情况 


VisualVM 是 Oracle JVM 自 带 的 可 视 化 工具 。 它 是 搬 件 架构 ， 采 用 标准 配置 ， 比 JConsole 用 起 
来 更 方便 。 

图 6-6 是 标准 的 VisualVM 汇 总 界面 。 启 动 VisualVM 并 把 它 连接 到 本 地 运行 的 程序 上， 就 能 看 
到 这 样 的 界面 。( VisualVM 也 能 连接 到 远程 应 用 上 ， 但 有 些 功能 通过 网 络 不 可 用 。) 这 个 界面 中 
VisualVM 连 接 的 是 MacBook Pro 上 运行 的 Eclipse， 你 可 以 看 到 我 们 用 来 编写 本 书 代码 的 Eclipse 的 
设置 。 图 6-6 如 下 所 示 。 


ee ptjaekra -By deep Rh i BF. et pe 
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图 6-6 VisualVM 汇 总 界面 


OD 在 Java 6 之 前 ， 加 镇 会 导致 一 次 原子 CAS ( Compare-And-Set ) 操作 。 对 于 没有 争 用 的 资源 ， 该 操作 会 造成 无 请 的 
开销 。 为 解决 这 一 问题 ，Java 6 中 引信 了 仿 癌 锁 技 术 ， 即 偶 问 于 第 一 个 加 镇 的 线程 ， 访 线程 后 续 加 钾 拘 作 不 青 要 
同步 。 其 基本 实现 方式 为 : 镇 最 初 为 NEUTRAL 状 态 ， 当 第 一 个 线程 加 锁 时 ,将 该 锁 的 状态 修改 为 BIASED， 并 记 
录 线 程 ID， 这 一 线程 在 后 续 加 锁 时 若 发 现状 态 是 BIASED 并 且 线 程 四 是 当前 线程 D， 则 只 设置 一 下 加 镇 标志 ， 不 
需要 进行 CAS 损 作 。 其 他 线程 关 要 加 这 个 镇 ， 需 要 使 用 CAS 损 作 将 状态 替换 为 REVOKE， 并 等 待 加 锁 标 志清 零 ， 
以 后 该 锁 的 状态 就 变 成 DEFAULT。 这 一 功能 可 用 -XxX: -UseBiasedLocking 作 令 禁 止 , 一 一 详 者 注 


内 自在 读书 他 
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右 侧 面板 顶部 有 很 多 标签。 其 中 有 扩展 ( Extension )、 样 例 ( Sampler )、JConsole 、MBeans 
和 VisualVM 捕 件 。VisualVM 搬 件 为 掌握 Java 运 行 时 的 动态 情况 提供 了 非常 棒 的 工具 。 建议 你 在 用 
VisualVM 做 任何 实际 工作 前 把 这 些 插件 都 装 上 。 
图 6-7 展 示 了 内 存 占 用 的 “ 饥 齿 ”模式 。 这 绝对 是 Java 平 台中 内 存 占用 情况 的 经 典 表现 。 它 表 
示 对 象 被 分 配 在 伊甸园 中 ， 使 用 ， 然 后 在 年 轻 代 中 被 回收 。 
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图 6-7 VisualVM 总 览 界面 


每 次 年 轻 代 收集 之 后 , 被 占用 的 内 存量 回落 到 基线 水 平 。 这 个 水 平 是 终身 制 对 象 和 幸存 者 对 
象 合 起 来 的 用 量 ， 可 以 用 它 来 确定 Java 进 程 的 健康 状况 。 如 果 基 线 在 进程 工作 时 保持 稳定 或 者 逐 
渐 北 减 ， 则 表明 内 存 的 使 用 情况 非常 健康 。 

如 果 基 线 水 平 上 升 ， 也 不 一 定 就 是 出 错 了 ,可 能 只 是 有 些 对 象 的 生存 期 很 长 , 长 到 足够 转 人 
终身 顾 养 园 中 。 在 这 种 情况 下 ， 最 终 会 进行 一 次 完全 收集 。 完 全 收集 会 导致 锯齿 模式 再 次 出 现 ， 
从 而 使 内 存 占用 回落 到 基线 水 平 。 如 果 完 全 收集 基线 持续 保持 稳定 ， 进 程 不 会 耗 光 内 存 。 

锯齿 上 和 斜坡 的 陡 度 是 进程 使 用 年 轻 代 内 存 ( 通常 是 伊 惫 园 ) 的 频率 ， 这 个 概念 很 重要 。 降 低 
年 轻 代 收 集 的 频率 基本 上 就 是 降低 锯齿 的 陡 度 。 

内 存 使 用 情况 的 另外 一 种 可 视 化 方式 如 图 6-8 所 示 。 你 能 看 到 伊甸园 、 幸存 者 乐园 (S0 和 S1 )、 
终身 磊 养 园 及 PermGen 区 。 在 程序 运行 时 ， 你 能 看 到 各 个 空间 的 大 小 变化 。 特 别 是 在 年 轻 代 收集 
之 后 ， 可 以 看 到 伊 乌 园 变 小 ， 幸 存 者 乐园 中 两 个 空间 的 角色 也 互相 转换 了 。 

探索 内 存 系统 和 运行 时 环境 有 助 于 你 理解 代码 如 何 运 行 。 相 应 地 ， 这 也 表明 VM 提供 的 服务 
对 性 能 影响 很 大 ， 所 以 你 绝对 应 该 花 时 间 研 究 一 下 VisualVM， 尤 其 要 结合 xmx 和 xms 这 些 选项 试 

-下 。 
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图 6-8 ”VisualVM 的 可 视 化 GC 插件 


下 一 节 中 , 我们 将 要 讨论 JVM 中 的 一 项 新 技术 , 这 项 技术 会 在 执行 过 程 中 自动 降低 堆 内 存 的 
占用 量 。 


6.5.7 人 狗 出 分 析 


本 市 介绍 了 JVM 最 近 的 一 项 修改 ,内容 仅 供 参考 ,程序 员 不 能 直接 控制 或 影响 这 项 修改 ,并 
且 在 最 近 发 布 的 Java 中 ， 这 项 优化 是 默认 的 。 因 此 本 节 中 没有 太 多 关于 这 项 修改 的 信息 或 例子 。 
所 以 如 果 你 想 了 解 一 下 JVM 提 升 自身 性 能 的 技巧 , 请 继续 。 如 果 没 兴趣 ， 可 以 跳 到 6.5.8 节 去 研究 
并 发 的 垃圾 收集 。 

锡 出 分 析 乍 一 看 是 个 相当 出 人 意料 的 想法 。 其 基本 思路 是 分 析 方 法 并 确认 其 中 哪个 局 部 变量 
(的 引用 类 型 ) 只 用 在 方法 内 部 ， 以 及 哪些 变量 不 会 传人 其 他 方法 或 从 当前 方法 中 返回 。 

这 样 JVM 就 可 以 在 当前 方法 的 栈 框架 内 部 创建 这 个 对 象 ， 而 不 再 使 用 堆 内 存 。 这 会 减少 程序 
年 轻 代 收 集 的 次 数 ， 从 而 提高 性 能 。 请 参见 图 6-9。 

这 就 是 说 可 以 避免 堆 分 配 ， 因 为 在 当前 方法 返回 时 ， 被 局 部 变量 占用 的 内 存 就 自动 释放 了 。 
用 这 种 不 牵扯 堆 分 配 的 方式 分 配 变量 空间 不 会 产生 垃圾 ， 当 然 就 不 需要 收集 垃圾 。 

逸 出 分 析 是 减少 JVM 垃 圾 收集 的 新 办 法 。 它 能 对 线程 的 年 轻 代 收集 次 数 产生 显著 影响 。 经 实 
践 证 明 ， 它 通常 能 对 总 体 性 能 产生 百 分 之 几 的 影响 。 虽 然 影响 不 是 特别 大 , 但 也 很 有 价值 ， 特 别 
是 在 进程 的 垃圾 收集 次 数 比 较 多 的 时 候 。 

从 Java 6u23 往 后 ， 逸 出 分 析 是 默认 打开 的 ， 所 以 新 版 Java 的 速度 免费 提升 了 。 
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仅 在 实现 多 出 分 
析 后 才 有 可 能 在 
栈 上 分 配 村 伙 


图 6-9 逸 出 分 析 避 人 免 了 对 象 的 堆 分 配 


现在 我 们 去 看 另外 一 个 对 代码 有 巨大 影响 的 环节 一 一 收集 策略 的 选择 ,我 们 从 一 个 经 典 的 高 
性 能 选择 (并 发 标记 清除 ) 开始 ， 然 后 看 一 看 最 新 的 收集 器 一 一 垃圾 优先 。 

选择 高 性 能 收集 器 有 很 多 原因 。 应 用 程序 可 能 会 从 较 短 的 GC 暂停 中 受益 ， 并 且 也 愿 蕊 运行 
更 多 线程 ( 占用 CPU 资源 ) 来 加 快速 度 。 或 者 你 想 控制 GC 和 暂停 的 频 度 。 除 了 基本 的 收集 器 ， 你 
还 可 以 用 选项 人 迫使 平台 采用 不 同 的 收集 策略 。 在 接 下 来 的 两 节 中 , 我 们 会 介绍 两 个 把 这 种 可 能 性 
变 成 现实 的 收集 器。 


6.5.8 并 发 标记 清除 


并 发 标记 清除 ( CMS ) 收集 器 是 Java 5 推荐 的 高 性 能 收集 器 , 在 Java 6 中 仍然 保持 了 旺盛 的 生 
命 力 。 可 以 通过 下 面 几 个 选项 激活 它 ， 如 表 6-4 所 示 。 
表 6-4 用 于 CMS 收集 器 的 选项 
选项 效果 _ 
-XX:+UseConcMarkSweepGc 打开 CMS 收 集 | 
-XX: +4CMSINncrementalMode 增 最 模式 (一 般 都 需要 ) 
-XX: +CMSIncrementalPacing 配合 增 量 模 式 , 根据 应 用 程序 的 行为 自动 调整 每 次 执行 的 
垃圾 回收 任务 的 幅度 【一般 都 需要 ) 
-XXK: +UseParNewGCc 并 爱 收 集 年 轻 民 
-XX:ParallelGCcThreads=<N> GCC 使 用 的 线程 数 


这 些 选 项 会 覆盖 垃圾 收集 的 默认 设置 ， 为 GC 配 置 有 N 个 并 行 线程 的 CMS 垃圾 收集 器 。 这 些 
线程 会 尽 可 能 地 在 并 发 模式 下 完成 GC 工 作 。 

这 种 并 发 方式 是 如 何 工 作 的 呢 ? 下 面 是 与 标记 清除 相关 的 三 个 重要 事实 ， 

口 某 种 世界 停 转 (简称 STW ) 的 暂停 是 不 可 避免 的 ; 

口 GC 子 系统 绝对 不 能 漏 邱 存活 对 象 ， 这 样 做 会 导致 IJVM 垮 掉 (或 者 更 精 ); 
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口 只 有 所 有 应 用 线程 都 为 整体 收集 暂停 下 来 ， 才 能 保证 收集 有 所有 的 垃圾 。 
CMS 利 用 了 最 后 一 点 。 它 制造 两 个 非常 短暂 的 STW 和 暂停 ， 并 且 在 GC 周期 的 剩余 时 间 和 应 用 

程序 的 线程 一 起 运行 。 这 表明 它 愿意 跟 “ 伪 阴性 ” 受 协 ， 由 于 竞争 危害 而 无 法 标识 某 些 垃圾 ( 被 

漏 掉 的 垃圾 会 在 下 一 个 GC 周 期 中 得 到 收集 ), 

CMS 还 要 在 运行 时 做 复杂 的 记 账 工作 , 记录 哪些 是 垃圾 , 哪些 不 是 。 这 些 额 外 的 开销 是 为 了 

在 不 俘 止 应 用 线程 的 情况 下 运行 GC 所 付出 的 代价 。CMS 在 有 更 多 CPU 核心 的 机 器 上 会 表现 得 更 

好 ， 并 且 会 制造 更 频繁 的 短暂 暂停 。 它 的 日 志 输 出 如 下 所 示 : 
2010-11-17T15:47:45.69240000: 90434 .570: [GC 90434.570: 

[ParNew: 14777K->14777K(14784K), 0.0000595 Becal90434 .570， 
[CMS: 1l114688K->114688K(114688K), 0.9083496 secs] 129465K->117349K(129472K), 


[CMS Perm : 49636K->49634K (65536K)] icms dc=100 ，D0.9086004 secs] 
[Times: uaer=0,.91 gys=0.00, real=0.91 secs] 


这 些 日 志和 6.4.4 节 中 基本 的 GC 日 志 差 不 多 ,但 增加 了 cMs 和 cMS Perm 收 集 器 部 分 。 

最 近 几 年 ，CMS 作 为 最 佳 高 性 能 收集 器 的 地 位 受到 了 挑战 ， 挑 战 者 是 垃圾 优先 ( G1 ) 收集 
器 。 我 们 来 看 看 这 颗 冉 内 升 起 的 新 星 ， 了 解 一 下 它 的 新 颖 方法 ， 以 及 它 能 够 突破 所 有 现存 的 Java 
收集 器 的 原因 。 


6.5.9 新 的 收集 器 : G1 


G1 是 Java 平 台中 身 新 的 收集 器 。 本 来 想 把 它 和 Java 7 一 起 发 布 , 但 后 来 作为 预 发 布 版 本 跟 Java 
6 一 起 发 布 了 ,到 Java 7 时 就 是 成 品 了 。 它 在 Java 6 中 并 没有 得 到 广泛 的 应 用 , 但 随 着 Java 7 逐渐 普 
肥 ， 有 和 望 让 G1 成 为 高 性 能 应 用 (也 可 能 是 所 有 应 用 ) 的 默认 选择 。 

G1 的 核心 思想 是 暂停 目标 ( pause goal )， 也 就 是 程序 在 执行 时 能 为 GC 暂停 多 长 时 间 ( 比如 
每 5 分 钟 20ms )。G1 会 竭尽 所 能 达成 暂停 目标 。 它 和 我 们 原来 遇 到 的 收集 器 完全 不 同 ， 并 且 开 发 
人 员 对 GC 如 何 执行 有 更 多 控制 权 。 

G1 不 是 套 正 的 分 代 式 垃圾 收集 器 ( 尽管 它 仍然 使 用 标记 清除 法 )。 相 反 ， G1 把 堆 分 成 大 小 相 
同 的 区 域 ( 比如 每 个 1 MB )， 不 区 分 年 轻 区 和 年 老区 。 暂 停 时 ， 对 象 被 撤 到 其 他 区 域 ( 就 像 伊 旬 
园 对 象 被 挪 到 幸存 者 乐园 一 样 )， 清 空 的 区 域 被 放 回 到 ( 空白 区 的 ) 自由 列表 上。 这 种 将 堆 划分 
为 大 小 相同 区 域 的 做 法 如 图 6-10 所 示 。 
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图 6-10 ”G1 如 何 划 分 堆 空 间 
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这 个 新 的 收集 策略 让 Java 平 台 可 以 统计 收集 单个 区 域 需 用 的 平均 时 长 。 这 样 你 就 可 以 在 合理 
范围 内 指定 一 个 暂停 目标 。G1 只 会 在 有 限 的 时 间 内 收集 尽 可 能 多 的 区 域 ( 尽管 在 收集 最 后 一 个 
区 域 时 所 用 的 时 间 可 能 比 预期 的 长 )。 

要 打开 G1， 需 要 用 刘表 6-$ 中 的 选项 。 
表 6-5 ”G1 收集 器 的 选项 
选项 效果 


-XX:+UseG1GC 打开 G1 收集 
-XX:MaxGCPauseMillis=50 告诉 G1 它 在 一 次 收集 中 暂停 的 时 间 应 读 尽 量 保 持 在 S0ms 以 内 
-XX:GCPauSsSeIntervalMillis=200 告诉 G1 它 将 两 次 收集 的 时 间 间 了 尽量 保持 在 200ms|! 以 上 


这 些 选 项 可 以 组 合 ， 比 如 设置 最 大 暂停 目标 是 50 ms， 暂停 间隔 不 能 少 于 200 ms。 当然，GC 
系统 所 能 承受 的 压力 是 有 限 的 。 必须 有 充足 的 暂停 时 间 把 垃圾 取出 来 。 每 隔 100 年 1ms 的 暂停 
肯定 是 不 现实 的 。 

G1 可 以 支持 的 负载 和 应 用 类 型 范围 很 广 。 如果 你 的 应 用 程序 已 经 到 了 需要 对 GC 调 优 的 地 步 ， 
G1 会 是 一 个 不 错 的 选择 。 

在 下 一 节 中 ， 我 们 会 介绍 JIT 编 详 。 对 于 很 多 或 大 多 数 程序 来 说 ， 这 是 唯一 一 个 可 以 为 产生 
高 性 能 代码 做 出 最 大 责 献 的 因素 。 我 们 会 学 习 JIT 编 译 的 基础 知识 ， 最 后 解释 一 下 如 何 打 开 JIT 编 
译 的 日 志 ， 让 你 能 够 判断 正在 编译 哪个 方法 。 


6.6 ” HotSpot 的 JIT 编译 


正如 我 们 在 第 1 章 所 讲 ，Java 是 一 种 “动态 编译 ”语言 。 也 就 是 说 在 程序 运行 时 ， 其 中 的 类 
还 会 再 进行 一 次 编译 ， 然 后 转换 成 机 岩 码 。 

这 个 过 程 称 为 即时 编译 或 JITing， 并 且 通 常 是 一 次 处 理 一 个 方法 。 要 在 庞大 的 代码 库 中 找 出 
其 中 的 重要 部 分 ， 理 解 这 个 过 程 是 关键 。 

下 面 是 一 些 与 JIT 编 译 有 关 的 基本 事实 。 

口 几乎 所 有 现代 JVM 中 都 有 某 种 JIT 编 译 器 。 

口 相 比 较 而 言 ， 纯 粹 解释 型 的 VM 要 慢 得 多 。 

口 编译 过 的 方法 在 运行 速度 上 要 比 解 释 型 的 代码 快 很 多 ， 非 常 多 。 

口 先 编译 用 得 最 多 的 方法 ， 这 是 有 道理 的 。 

口 在 做 JIT 编 译 时 ， 先 处 理 唾 手 可 得 的 编译 很 重要 。 

按照 最 后 一 点 ， 我 们 应 该 先 研 究 编译 过 的 代码 ,因为 在 正常 情况 下 , 所 有 仍然 处 于 解释 状态 
下 的 方法 都 设 有 已 经 编译 过 的 方法 运行 频繁 。 偶 尔 会 有 无 法 编译 的 方法 ， 但 非常 罕见 。 

方法 一 开始 都 是 以 字 节 码 形 态 存 在 的 ,， 有 调用 时 JVM 只 会 对 字 节 码 进行 解释 并 执行 ， 同时 记 
录 方 法 被 调用 的 次 数 及 其 他 一 些 统计 数据 。 当 被 调用 次 数 达 到 某 个 疮 值 ( 默认 10 000 次 ) 后 ， 如 


1 入 
和 


6.6 HotSpot 的 JIT 编 译 163 


果 它 是 合格 的 方法 ,就 会 有 个 JVM 线 程 在 后 台 把 它 的 字 节 码 编译 成 机 器 码 。 如 果 编译 成 功 ， 以 后 
所 有 对 该 方法 的 调用 都 会 用 它 的 编译 结果 ,除非 出 现 了 某 些 导致 检验 失效 的 情况 ,或 者 出 现 了 北 
优化 ”。 

根据 实际 情况 ， 方 法 编译 后 产生 的 机 器 码 运行 速度 可 能 比 解释 模式 下 的 字 节 码 快 100 倍 。 改 
善 性 能 通常 都 要 先 弄 明白 程序 中 哪些 方法 比较 重要 ， 以 及 哪些 重要 的 方法 被 编译 了 。 
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i ee 和 [ en ST 了 胆 re a E rr i 
2 = ha Pa -~ EF 
I Pb 本 | a L 站 


一 为 什么 不 提前 编译 好 ( 像 C++ 一 

样 )。 第 一 个 答案 通常 都 是 : 因为 用 平台 无 美的 东西 ( .jar 和 .class 文 件 ) 作为 基本 部 署 单位 要 比 
为 每 个 目标 平台 做 一 份 不 同 的 编译 好 的 二 进 制 文件 更 轻松 。 

另外 一 种 答案 是 动态 编译 会 给 编译 器 提供 更 多 信息 。 具体 地 说 ， 提前 ( AOT ) 编译 的 语言 


eg 问 ， Java 平 台 台 为 什 乏 PE 心 ;去 做 动 态 编译 一 


得 不 到 运行 时 的 任何 信息 一 一 比如 某 个 指令 是 否 可 用 ,其 他 的 硬件 细节 以 及 代码 运行 情况 的 统 
计数 据 。 这 些 变数 让 事情 变 得 很 有 趣 ， 使 得 Java 这 样 的 动态 编译 语言 实际 上 可 能 会 比 提前 编译 
的 语言 运行 得 更 快 。 


在 接 下 来 对 JITing 机 制 的 讨论 中 ,我 们 所 说 的 JVM 特 指 HotSpot。 后续 讨 论 中 很 多 通用 内 容 也 
适用 于 其 他 VM， 但 在 具体 细节 上 可 能 会 有 很 大 出 人 。 

我 们 会 先 介绍 一 下 HotSpot 提 供 的 几 个 JIT 编 译 器 , 然后 解释 HotSpot 中 最 有 力 的 两 项 优化 技术 
( 内 联 和 独占 派发 )。 在 本 节 的 结尾 , 我 们 会 告诉 你 如 何 打 开 方 法 编译 日 志 ， 以 便 你 可 以 看 到 被 编 
译 的 确切 方法 。 下 面 有 请 HotSpot。 


6.6.1 介绍 HotSpot 


Oracle 收 购 Sun 时 拿 到 了 HotSpot VM ( 原来 收购 BEA 时 还 拿 到 一 个 J 玉 ockit )。HotSpot 是 
OpenJDK 的 基础 。 它 有 两 种 运行 模式 一 一 客户 端 模式 和 服务 屁 端 模式 。 可 以 在 启动 JVM 时 指定 
-client 或 -server 选 项 来 选择 不 同 的 模式 。( 必须 是 命令 行 中 的 第 一 个 选项 ) 每 种 模式 都 有 各 
自 适 用 的 应 用 程序 。 

1. 客 尸 端 编译 器 

客户 端 编译 器 主 要 用 于 GUI 应 用 程序 。 在 这 个 领域 中 ， 操 作 的 一 致 性 至 关 重 要 ， 所 以 客户 端 
编译 器 ( 有 时 叫 Cl ) 在 编译 时 所 做 的 决定 往往 更 保守 。 也 就 是 说 它 不 能 因为 要 取消 一 个 经 证 实 不 
正确 或 基于 错误 假设 的 优化 决定 而 意外 暂停 。 

2. 服务 器 端 编译 器 

相反 ， 服 务 佣 上 问 编 详 器 〈(C2 ) 在 编译 时 会 大 胆 假设 。 为 了 确保 代码 正确 运行 ，C2 会 快速 地 


册 JVM 的 动态 优化 技术 可 能 会 基于 一 些 大 胆 ( 甚至 不 安全 ) 的 假设 来 编译 字 节 码 。 比 如 假定 要 处 理 的 数据 都 属于 某 
一 类 ， 而 在 编译 结果 中 只 保留 处 理 该 类 数据 的 程序 分 支 。 如 果 假 设 不 成 立 ， 则 JVM 只 能 放弃 编译 结果 ， 回 去 解释 
并 执行 原来 的 字 节 码 ， 这 一 过 程 被 称 为 逆 优 化 。 一 一 译 者 广 
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做 一 次 运行 时 检查 (通常 被 称 为 警戒 条 件 )， 以 确保 假设 有 效 。 如 果 假 设 无 效 ， 它 会 取消 这 次 编 
译 ， 并 尝试 别 的 编译 。 这 种 大 胆 假设 的 方式 比 保守 的 客户 端 编译 器 产生 的 编译 结果 性 能 好 很 多 。 

3. 实时 Java 

近年 来 出 现 了 一 种 实时 Java 平 台 ， 有 些 开 发 人 员 好 奇 为 什么 那些 需要 表现 出 高 性 能 的 代码 不 
直接 用 这 个 平台 ( 它 是 独立 的 JVM， 不 是 HotSpot 选 件 ) 那 是 因为 实时 系统 不 一 定 是 最 快 的 。 

实时 编程 的 关注 点 实际 上 是 承诺 能 否 兑现 。 从 统计 和 角度 讲 , 实时 系统 是 为 了 让 执行 操作 的 时 
间 尽 量 保持 一 致 , 并 且 为 了 达成 这 个 目的 , 它 可 能 会 牺牲 一 些 平均 等 等 时 间 。 为 了 让 运行 状况 保 
持 一 致 ， 整 体 性 能 是 可 以 受到 轻微 影响 的 。 

图 6-11 中 有 两 组 代表 等 待 时 间 的 点 阵 。 系 列 2 (上面 那 组 点 阵 ) 的 平均 等 待 时 间 在 增长 ( 因 
为 它 的 等 待 时 间 刻 度 更 高 ), 但 方差 在 减 小 ， 因 为 这 些 点 比 系列 1 中 的 点 更 靠近 自己 的 平均 值 ， 系 
列 1 的 点 阵 相 较 而 言 分 布 更 加 广泛 。 
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但 硕 望 实现 高 性 能 表现 的 团队 想 要 的 是 更 低 的 平均 等 待 时 间 , 即便 以 更 高 的 方差 为 代价 ,所 
以 他 们 通常 会 选择 服务 器 请 编 详 天 的 大 胆 优化 策略 ( 对 应 系列 1 ) 

接 下 来 我 们 会 讨论 所 有 运行 时 ( 服务 器 端 、 客 户 端 和 实时 ) 广泛 采用 的 技术 ,这 项 技术 使 它 
们 表现 得 更 好 。 


6.6.2 ”内 联 方法 
内 联 是 HotSpot 的 最 大 卖点 之 一 。 凡 联 的 方法 不 再 是 被 调用 ， 而 是 将 调用 方法 的 代码 直接 放 
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到 调用 者 内 部 。 

平台 有 这 方面 的 优势 , 编译 器 可 以 根据 运行 时 的 统计 数据 ( 方法 的 调用 频率 ) 和 其 他 因素 ( 比 
如 会 不 会 因为 调用 者 方法 太 多 而 对 代码 缓存 产生 影响 ) 来 决定 如 何 处 理 内 联 。 也 就 是 说 HotSpot 
编译 右 所 做 的 内 联 决策 比 提前 编译 的 编译 器 更 智能 。 

方法 的 内 联 是 完全 自动 的 , 并 且 默 认 参 数值 几乎 适用 于 任何 情况 。 但 也 有 选项 可 以 用 来 控制 
内 联 方法 大 小 ， 以 及 方法 在 成 为 内 联 候选 之 前 的 调用 频率 要 达到 多 高 。 对 于 好 奇 的 程序 员 来 说 ， 
这 些 选项 对 于 深入 了 解 内 联 如 何 工作 很 有 儿 助 。 通常 它 们 对 于 生产 环境 下 的 代码 用 处 不 大 , 并 且 
应 该 作为 性 能 调 优 的 最 后 选择 ， 因 为 它们 对 运行 时 系统 的 性 能 可 能 存在 不 可 预测 的 影 啊 。 
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pp pp re rr 
他 们 认为 变量 是 私有 的 , 方法 调用 不 能 因为 优化 而 去 掉 ， 不 能 在 类 外 访问 这 个 变量 。 这 种 想法 
不 对 。HotSpot 把 方法 编译 成 机 器 码 时 能 够 并 且 会 息 略 访 问 控制 ， 不 用 访问 器 方法 直接 访问 私 
有 域 。 这 并 不 违背 Java 的 安全 模型 ， 因为 所 有 访问 控制 都 在 类 加 载 和 连接 阶段 检查 过 。 

如 果 你 还 不 信 ， 可 以 做 个 练习 ， 写 一 个 跟 代 码 清 单 6-2 类 人 
访问 器 方法 的 速度 和 直接 访问 公共 域 的 速度 。 


6.6.3 动态 编译 和 独占 调用 


独占 调用 就 是 这 种 大 胆 优化 的 例子 之 一 。 它 是 基于 大 量 观察 做 出 的 优化 , 像 下 面 这 种 对 象 上 
的 方法 幸 用 ; 

MYyRaActtualClasSSNoOtInEertacCe obj = getIinstance(); 

obij.callMyMethod'(), 


只 会 在 一 种 类 型 的 对 象 上 调用 。 换 句 话 说， 就 是 调用 点 obj .callMyMethod|) 几乎 不 会 同 
时 碰 到 一 个 类 和 它 的 子 类 。 这 时 可 以 把 Java 方 法 查找 替换 为 callMyMethoad1() 编译 结果 的 直接 
调用 。 


提示 独占 派发 提供 了 一 个 剖 析 JVM 运 行 时 的 例子 ， 殉 许 Java 平 各 台 ;进行 CH+ 这 种 AOT 语 计 实现 不 
了 的 优化 。 


出 于 非 技 术 的 原因 ，getInstance() 方 法 有 时 不 能 返回 MyActualclassNotInterface 类 
型 的 对 象 ， 而 其 他 情况 下 不 能 返回 一 些 子 类 的 对 象 , 但 实际 上 这 种 情况 几乎 从 没 发 生 过 。 但 为 了 
防止 这 种 情况 出 现 , 会 有 一 个 运行 时 检查 来 确保 对 象 的 类 型 是 由 编译 器 按 预期 插入 的 。 如 果 这 个 
预期 被 违背 ， 运 行 时 会 取消 优化 ， 程 序 甚至 都 不 会 注意 也 不 会 犯 任何 错误 。 

只 有 服务 各 名 编 详 融 才 会 做 这 种 大 胆 的 优化 。 实 时 和 客户 端 编译 器 都 不 会 这 样 做 。 
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6.6.4 ” 读 懂 编译 日 志 


我 们 来 看 一 个 例子 ， 了 解 一 下 如 何 使 用 JIT 编 译 日 志 。 依 巴 谷 星 表 中 详细 列 出 了 从 地 球 上 可 
以 观测 到 的 星星 。 我 们 的 程序 会 处 理 这 个 目录 ,产生 能 在 指定 夜晚 、 指 定 地 址 看 到 的 星 图 。 

我 们 来 看 这 个 程序 输出 的 一 些 日 志 , 看 看 在 星 图 应 用 运行 时 编译 了 哪些 方法 。 我 们 用 的 关键 
选项 是 -xxX:+Printcompilation。 我 们 前 面 简单 讨论 过 这 个 扩展 选项 。 把 这 个 选项 加 到 局 动 
JVM 的 命令 里 是 告诉 JIT 编 译 线程 输出 标准 日 志 。 这 些 日 志 表 明 方 法 超过 编译 国 值 并 被 转 成 机 莫 


码 的 时 间 。 

1 java.lang.String: :hashcode (64 bytes) 

2 java.math.BigInteger: :mulAdd (81 bytes) 

3 Java.math.BigInteger: :mUJtiplyToLen (219 bytes) 

4 java.math.BigIntegqer: :addone (77 bytes) 

5 java.math.BiglInteger: :sgquareToLen (172 bytes) 

6 java.math.BigqgInteger: :primitiveLeftShift (79 bytes) 

7 Java.math.BigInteger: :montReduce (99 bytes) 

8 SUN. Security.provider .SHA: :implCompress (491 bytes) 

9 java.lang.string: :charAt (33 byteas) 

i ! sun.nio.cs.SingleByteDecoder: :decodeArrayLoop @ 129 (308 bytes) 
39 sun.misc.FloatingDecimal: :doubleValue (1269 bytes) 
440 org.camelot .hipparcos .DelimitedLine: :getNextString (5 bytes) 
1 org.camelot .hipparcos .Star: :parsestar (301 bytes) 

2 |! org.camelot .CamelotSstarter: :populateStarSstore @ 25 (106 bytes) 
65 如 java.lang.stringBuffer: :append (8 bvytes) 


这 是 非常 由 型 的 Printcompilation 输 出 。 这 些 日 志 表 明了 “ 热 ” 到 可 以 编译 的 方法 。 跨 你 
想 的 一 样 ， 第 一 个 被 编译 的 方法 很 可 能 是 平台 方法 ( 比如 String#hashcode )。 再 过 一 段 时 间 ， 
应 用 方法 ( 比如 org.camelot.hipparcos.Scar#parsescar 方 法 ， 在 例子 中 用 于 分 析 天 文 目 
录 里 的 记录 ) 也 会 被 编译 。 

这 些 输出 中 每 行 都 有 个 数字 ,表明 了 这 些 方法 在 这 次 运行 中 的 编译 顺序 。 注 意 ， 由 于 平台 和 
动态 性 质 ， 这 个 顺序 在 每 次 运行 时 可 能 会 簿 有 变化 。 这 里 还 有 一 些 其 他 域 。 

明 该 方法 是 同步 的 。 


口 $ 一 一 当前 栈 替 换 ( OSR )。 这 个 方法 被 编译 了 ， 并 且 换 掉 了 运行 代码 中 的 解释 型 版 本 。 
注意 ，OSR 方 法 有 它们 自己 的 计数 方案 ， 从 1 开始 。 

小 心 人 僵尸 

当 查 看 用 服务 器 端 编译 器 ( C2 ) 运 行 代码 的 样 例 日 志 时 , 你 可 能 偶尔 会 看 到 “ 变 得 无 法 进 人 ” 
和 “ 变 成 僵尸 ”这 样 的 字眼 。 这 表明 由 于 类 加 载 操 作 (通常 情况 下 )， 某 个 已 经 被 编译 过 的 特定 
方法 现在 无 效 了 。 

逆 优 化 

如 果 经 证 实 代 码 优 化 所 基于 的 假设 是 不 真实 的 ，HotSpot 可 以 对 代码 进行 逆 优 化 操作 。 在 许 
多 情况 下 ， 它 会 重新 考虑 ， 党 试 不 同 的 优化 。 因 此 同一 个 方法 可 能 会 被 道 优化 和 重 编译 几 次 。 
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过 了 一 段 时 间 , 你 会 看 到 被 编译 的 方法 数量 趋 于 稳定 ,编译 好 的 代码 达到 了 一 个 稳定 的 状态 ， 
并 且 大 多 数 代 码 会 保持 不 变 。 哪 些 方法 被 编译 取决 于 所 用 的 JVM 版 本 和 OS 平台 。 并 不 是 所 有 平 
台 都 会 产生 相同 的 编译 方法 集合 , 并 且 给 定 方法 编译 代码 的 大 小 也 不 会 完全 一 样 。 就 像 性 能 调 优 
里 很 多 其 他 东西 一 样 , 这 也 应 该 进行 测量 , 并 且 结 果 可 能 会 让 人 大 吃 一 惊 。 即 便 看 上 去 相当 简单 
的 Java 方 法 ， 在 Solaris 和 Linux 上 经 JIT 编 译 生 成 的 机 器 人 码 也 会 有 五 分 之 一 的 差异 。 测 量 是 必 不 可 
少 的 。 


6.7 小结 


性 能 调 优 不 是 盯 着 你 的 代码 期 待 奇迹 , 或 者 给 代码 喝 一 雏 快 速 修复 药水 。 相 反 , 性 能 调 优 需 
要 细致 测量 ， 关 注 细节 ,还 需要 你 的 耐心 。 你 要 不 断 减 少 测试 中 出 现 的 错误 源 ， 直 到 引发 性 能 问 
题 的 真正 凶手 出 现 。 

我 们 先 来 看 看 在 JVM 动 态 环 境 中 进行 性 能 调 优 的 要 点 。 

口 JVM 是 极为 强大 的 复杂 运行 时 环境 。 

口 JVM 的 性 质 使 得 有 时 候 优 化 其 中 的 代码 很 有 挑战 性 。 

口 你 必须 通过 测量 准确 地 找到 问题 的 真正 所 在 。 

口 要 特别 注意 垃圾 收集 子 系统 和 JIT 编 译 问 。 

口 监测 还 有 其 他 一 些 工具 对 你 真 的 很 有 帮助 。 

口 学 会 阅读 日 志和 平台 的 其 他 指标 一 一 有 时 不 能 使 用 工具 。 

口 你 必须 测量 并 设置 目标 (这 个 太 重 要 了 ， 所 以 我 们 要 一 再 提起 )。 

现在 你 应 该 具备 探索 和 实验 Java 平 台 的 高 级 性 能 特性 所 需 的 基础 知识 了 , 并 且 能 够 理解 性 能 
机 制 如 何 影响 你 的 代码 。 希望 你 能 开放 心态 ， 以 足够 的 信心 和 经 验 去 分 析 这 些 数据 ， 并 能 把 这 种 
见解 应 用 于 你 自己 的 性 能 问题 。 

我 们 会 在 下 一 章 看 到 JVM 上 除 Java 语 言 之 外 的 其 他 语言 , 平台 的 很 多 性 能 特性 适用 范围 非常 
广泛 一 一 特别 是 JIT 编 译 器 和 GC 的 相关 知识 ，。 
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这 一 部 分 专门 探索 JVM 上 的 新 语言 范式 和 多 语言 编程 。 

JVM 是 一 个 迷人 的 运行 时 环境 ; 它 提供 的 不 仅 是 性 能 和 能 力 ,还 赋予 了 程序 员 惊 人 的 灵活 性 。 
实际 上 ，JVM 是 探索 Java 之 外 的 语言 的 关口 , 并 且 会 让 你 尝试 一 些 不 同 的 编程 方式 。 

如 果 你 只 用 Java 写 过 程序 , 可 能 相知 道学 习 其 他 语言 会 有 什么 好 处 。 就 像 我 们 在 第 1 章 说 的 ， 
成 为 优秀 Java 开发 人 员 的 本 质 就 是 对 Java 语言 、 平 人 台 和 生态 系统 的 方方面面 掌握 得 越 来 越 全 面 . 
这 包括 能 够 欣赏 那些 目前 刚刚 起 步 ， 但 不 义 的 将 来 就 会 变 得 不 可 或 缺 的 主题 。 

未 来 已 经 发 生 ， 只 是 分 布 尚 不 均匀 。 

-一 一 威 靡 . 吉布森 

事实 证 明 ， 很 多 未 来 需要 的 新 想法 已 经 出 现在 国 数 式 编 程 等 其 他 JVM 语言 中 了 。 学 习 新 
JVM 语言 的 过 程 中 ， 我 们 可 L 一 管 男 一 个 世界 ， 我 们 未 来 的 某 些 项 目 很 可 能 就 跟 它 很 像 。 从 不 
同 的 视角 探索 问题 能 帮 我 们 重新 审视 已 有 的 知识 。 学 习 新 语言 可 以 开启 新 的 可 能 性 ， 我 们 可 能 
会 发 现 自己 不 知道 的 新 天 赋 ， 泡 提 新 技能 ， 而 这 些 东 西 总 有 一 天 会 派 上 用 场 . 

第 7 章 会 解释 一 下 为 什么 Java 不 是 解 块 所 有 器 题 的 理想 语言 、 为 什么 国 数 式 编 程 概 念 有 用 ， 
以 及 如 何 为 特定 项 目 选 择 一 种 疾 Java 语言 。 

最 近 ， 很 多 书 和 博客 里 都 提出 一 种 观点 ， 认 为 函数 式 编程 很 快 就 会 成 为 每 个 开发 人 员 职 业 
生涯 中 的 重要 和 角色。 很 多 文章 都 把 函数 式 编 程 描述 得 令 人 生 上 县 ， 却 常常 讲 不 清楚 钞 数 式 编 程 怎 
么 在 Java 这 样 的 语言 中 “发 光 发 热 ” 

实际 上 ， 函 数 式 编程 根本 算 不 上 一 个 整体 结构 。 相 反 ， 它 更 像 一 种 风格 ， 开 发 人 员 思 考 方 
式 上 的 一 个 过 流 。 第 8 章 会 给 出 一 个 用 Groovy 语言 编写 的 、 稍微 带 点 儿 函 数 式 编程 味道 的 例子 ， 
就 是 用 一 种 更 清晰 的 、 不 太 容 易 出 bug 的 风格 来 处 理 集合 的 代码 。 在 第 9 章 ， 我 们 会 用 Scala 语 
言 讨论 对 象 一 函数 式 风 格 。 第 10 章 会 用 Clojure 语言 看 一 下 纯粹 的 国 数 式 编程 〈 它 甚至 超过 了 
面 问 对 象 ) 方式 。 

在 第 四 部 分 , 我 们 会 介绍 几 个 真实 案例 , 针对 这 些 案例 , 其 他 语言 能 够 给 出 更 好 的 解决 方案 。 
如 果 你 不 信 ， 可 以 提前 看 一 下 第 四 部 分 ， 然 后 再 回来 学 习 应 用 那些 技术 所 需 的 语言 。 
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本 章 内 容 

D 为 什么 应 该 使 用 备 选 JVM 语 计 
D 语言 的 类 型 

D 备 选 语言 的 选择 标准 

口 JVM 如 何 处 理 备 选 语言 


如 果 你 用 Java 做 过 大 项 目 ， 可 能 已 经 注意 到 了 ，Java 有 时 稍 显 繁琐 和 笔 拙 。 你 甚至 可 能 希望 
它 不 是 这 样 的 一 一 总 之 要 再 容易 点 儿 。 

好 在 JVM 很 棒 ! 实际 上 ， 它 太 棒 了 ，Java 以 外 的 其 他 语言 也 可 以 很 自然 地 把 它 当 成 栖息 地 。 
我 们 在 本 童 里 会 告诉 你 为 什么 要 把 其 他 JVM 编 程 语 言 加 入 到 我 们 的 项 目 中 ， 以 及 如 何 做 到 这 
一 成 6 

我 们 会 讨论 描述 不 同 语 言 类 型 ( 比如 静态 与 动态 ) 的 方式 、 为 什么 用 备 选 语言 ， 以 及 选择 它 
们 时 有 哪些 标准 。 我 们 还 会 介绍 三 种 语言 ，Groovy 、Scala 和 Clojure， 并 在 第 三 部 分 和 第 四 部 分 
中 更 深入 地 探讨 它们 。 

然而 在 开始 之 前 ， 你 需要 对 Java 的 缺点 有 更 清楚 的 认识 。 下 一 节 有 一 个 扩 
Java 语 言 中 一 些 恼 人 的 地 方 ， 指 出 了 它 未 来 的 发 展 方向 为 函数 式 编程 风格 。 


7.1 Java 太 答 ? 纯粹 诽谤 


假设 你 要 在 一 个 交易 ( 事务 ) 处 理 系统 中 编写 一 个 新 组 件 。 这 个 系统 的 简化 视图 如 图 7-1 
所 示 。 

在 图 中 可 以 看 到 ， 系 统 有 两 个 数据 源 : 上 游 的 收 单 系统 ( 可 以 通过 Web 服 务 查询 ) 和 下 游 的 
派发 数据 库 。 

这 是 一 个 很 现实 的 系统 ,是 Java 开 发 人 员 经 常 构建 的 系统 。 我 们 在 这 一 节 里 准备 引入 一 小 自 
代码 把 两 个 数据 源 整合 起 来 。 你 会 看 到 Java 解 决 这 个 问题 有 点 生 拙 。 之 后 我 们 会 介绍 函数 式 编程 
的 一 个 核心 概念 。 并 展示 一 下 怎么 用 映射 (map ) 和 过 滤器 (filter ) 等 函数 式 特性 简化 很 多 常见 
的 编程 任务 。 你 会 看 到 Java 由 于 缺乏 对 这 些 特性 的 直接 支持 ， 编 程 会 困难 不 少 。 


展示 例 ， 它 突出 了 
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图 7-1 ”交易 处 理 系统 的 例子 


7.1.1 整合 系统 


我 们 需要 一 个 整合 系统 来 检查 数据 确实 到 了 数据 库 。 这 个 系统 的 核心 是 reconcile() 方 法 ， 
它 有 两 个 参数 ，sourceData (来 自 于 Web 服 务 的 数据 ， 归 结 到 一 个 Map 中 ) 和 dbIas。 


你 需要 从 sourceDpata 中 取出 main_ref 键 值 ， 用 它 跟 数据 库 记 录 的 主键 比较 。 代 码 清单 7-] 
是 进行 比较 的 代码 。 


代码 清单 7-1 整合 两 个 数据 源 : 
public void reconcile (lList<Map<String, String»» sourceData, 
Set<String> dbIds) | 
Set<String> seen = new HashSet <String>(); 
MAIN: for (Map<String, String> row : SourceData) { 
string pTradeRef = row.get ("main ref"),; 


| 假定 pTraadeRef 永 远 
if (dbIds.contains (pTradeRef)) | 不 会 为 mul1 


System.out .println (pTradeRef +" OR"); 
Seen.add (pTradeRef ); 
} else 1 


System.out .printlnl"main ref: "+ pTradeRef +" not present in DB"), 


} 


特殊 情况 
for (String 上 ia : dbIds) 1 a 


if {lgeen.containg (tid)) 1 
syatem.out .printlnl"main ref: "+ 上 la +" seen in DB but not Sourcen) 
| 


| 
} 
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这 里 主要 是 检查 收 单 系统 中 的 所 有 订单 是 否 都 出 现在 派发 数据 库 里 。 这 项 检查 由 打上 了 
MAIN 标 签 的 for 循环 来 做 。 

还 有 另外 一 种 可 能 。 比 如 有 个 实习 生 通 过 管理 界面 做 了 些 测试 订单 (他 没 意 识 到 这 些 订单 用 
的 是 生产 系统 )。 这样 订 单数 据 会 出 现在 派发 数据 库 里 ， 但 不 会 出 现在 收 单 系统 中 。 

为 了 处 理 这 种 特殊 情况 ， 还 需要 一 个 循环 。 这 个 循环 要 检查 所 见 到 的 集合 ( 同时 出 现在 两 个 
系统 中 的 交易 ) 是 否 包 含 了 数据 库 中 的 全 部 记录 。 它 还 会 确认 那些 遗漏 项 。 下 面 是 这 个 样 例 的 一 
部 分 输出 : 

T172329 OK 

1REGY OFK 

lRGGHW OK 

main ref: 1R6H2 not present in DB 

main ref: lR6H3 not present in DE 

1R6HE OK 

哪儿 出 错 了 ? 原来 是 上 游 系统 不 区 分 大 小 写 而 下 游 系统 区 分 , 在 派发 数据 库 里 表示 为 1R6H12 
的 记录 实际 上 是 1r6h2。 

如 果 你 检查 一 下 代码 清单 7-1， 就 会 发 现 问题 出 在 contains () 方 法 上 。contains () 方 法 会 
检查 其 参数 是 否 出 现在 目标 集合 中 ， 只 有 完全 匹配 时 才 会 返回 true。 

也 就 是 说 其 实 你 应 该 用 containscaseInsensitive1l) 方 法 ， 可 这 是 一 个 根本 就 不 存在 的 
方法 ! 所 以 你 必须 把 下 面 这 段 代 码 

if tdbIds.contains (pTradeRef)) | 

System.ourt .println(pTradeRef +" OK"),; 
seen.add(lpTradeRef); 


| elae | 
System.out .println("main ref: "+ pTradeRef +" not present in DB”) ; 


换 成 这 样 的 循环 : 
for (string id : dbIds) 1 
if (id.equalsIgnoreCase [pTradeRef})) | 

Syastem.out .println(lpTradeRef +" OR"),; 

Seen .add (pTradeRef).; 

continue MAIN; 
| 
system.out .printlin(nmain ref: "t prradeRef +" not present in DB"); 
这 看 起 来 比较 笨重 。 只 能 在 集合 上 执行 循环 操作 , 不 能 把 它 当 成 一 个 整体 来 处 理 。 代 码 既 不 

简 请 ， 又 似乎 很 脆弱 。 

随 着 应 用 程序 逐渐 变 大 ， 简 洁 会 变 得 越 来 越 重 要 一 一 为 了 节约 脑力 ， 你 需要 简洁 的 代码 。 


7.1.2 ”函数 式 编程 的 基本 原理 


希望 上 面 的 例子 中 的 两 个 观点 引起 了 你 的 注意 
口 将 集合 作为 一 个 整体 处 理 要 比 循环 遍历 集合 中 的 内 容 更 简洁， 通常 也 会 更 好 。 
口 如 果 能 在 对 和 象 的 现 有 方法 上 加 一 点 点 远 辑 来 调整 它 的 行为 是 不 是 很 棱 呢 ? 


a 
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如 果 你 遇 到 过 那 种 基本 就 是 你 需要 , 但 又 稍微 差点 儿 意思 的 集合 处 理 方法 ,你 就 明日 不 得 不 
再 写 一 个 方法 是 多 么 诅 南 了 ， 而 函数 式 编程 ( FP ) 恰好 拖 到 了 这 个 痒 处 。 

换 种 说 法 , 简洁 (并 且 安 全 ) 的 面向 对 象 代码 的 主要 限制 就 是 ， 不 能 在 现 有 方法 上 添加 额外 
的 逻辑 。 这 将 我 们 引 向 了 FP 的 大 思路 :假定 确实 有 办 法 向 方法 中 添加 自己 的 代码 来 调整 它 的 功能 。 

这 意味 着 什么 7 要 在 已 经 固定 的 代码 中 添加 自己 的 处 理 逻 辑 , 就 需要 把 代码 块 作为 参数 传 到 
方法 中 。 下 面 这 种 代码 才 是 我 们 真正 想 要 的 (为 了 突出 ， 我 们 把 这 个 特殊 的 contains () 方 法 加 
粗 了 ); 


if (dbIds.contains (pTradeRef, matchFunction)) | 
System.out .printlnlpTradeRef +" OK"); 
seen. add lpTradeRef); 
} else | 
System.out printlnl"main ref: "+ pTradeRef +" not present in DB"); 


如 来 能 这 样 瑟 ，contains() 方 法 就 能 做 任何 检查 ， 比 如 匹配 区 分 大 小 写 。 这 需要 能 把 匹配 
哺 数 表示 成 值 ， 即 能 把 一 段 代码 写成 “函数 字面 值 ”并 赋值 给 一 个 变量 。 

明 数 式 编程 要 把 逻辑 ( 一 般 是 方法 ) 表示 成 值 。 这 是 FP 的 核心 思想 , 我们 还 会 再 次 讨论 ， 先 
看 一 个 带 点 儿 FP 思想 的 Java 例 子 。 


7.1.3 ”映射 与 过 滤器 
我 们 把 例子 稍微 展开 一 些 ， 并 放 在 调用 reconcile() 的 上 下 文中 : 


reconcilelsourceData, new HashSet<String> (extractPrimaryKeys (dbIinfos))):; 
private List<String> extractPrimarykeys (List<DBInfo> dbInfos) 1 
List<String» out = new ArrayList<>|(); 
for (DBInfo tinfo ; dbinfos) | 
out .add (tinfo.primary key); 
} 
return out ; 
| 
extractPrimaryKeys() 方 法 返回 从 数据 库 对 象 中 取出 的 主键 值 (字符 串 ) 列表 。FP 粉 管 
这 叫 map () 表达 式 ， extractPrimaryKeys() 方 法 按 豚 友 处 理 List 中 的 每 个 元 率 ， 然后 百合 返回 
一 个 List。 上 面 的 代码 构建 并 返回 了 一 个 新 列表 。 
注意 ， 返 回 的 List 中 元 素 的 类 型 ( string ) 可 能 跟 输 入 的 List 中 元 素 的 类 型 (DBInfo ) 
不 同 ， 并 且 原 始 列表 不 会 受到 任何 影响 。 
这 就 是 “函数 式 编程 ”名 称 的 由 来 ， 函 数 的 行为 跟 数学 函数 一 样 。 函 数 f (x) =x*x 不 会 改变 
输入 值 2， 只 会 返回 一 个 不 同 的 值 4。 


pe 和 eo WE El a ES ee | 3 了 Py 的 优 ye 2 ee 本 寺中 : el i 
a a rp yea , yKeys () 返回 的 
List 传 入 HashSet 构 造 方法 中 ， 变 成 Set。 这 样 可 以 去 掉 List 中 的 重复 元 素 ，reconcile() 
方法 调用 的 contains () 可 以 少 做 一 些 工 作 。 
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map ( ) 是 经 典 的 FP 惯用 语 。 它 经 常 和 另 一 个 知名 模式 成 对 出 现 : filter () 形 态 ， 请 看 代码 
清单 7-2。 


代码 清单 7-2 ”过 滤器 形态 | | 
rt et String>» TR 全 示 癌 必 二 Ma 辣 二 这 三 全 string>> in) 1 
List<Map<String, String>> out = new ArrayList<>|(); 
for (Map<String, String> msg : in) 1 | 肪 和 性 复制 
if (imsg.get ("status") .equalsIgnoreCase ("CANCELLED")) | 
out .add (msg) } 
3 
} 
retuwrn out.; 
} 
注意 其 中 的 防御 性 复制 ， 它 的 意思 是 我 们 返回 了 一 个 新 的 List 实 例 。 这 段 代码 没有 修改 原 
有 的 List ( filter () 的 行为 跟 数 学 函数 一 样 )。 它 用 一 个 函数 测试 每 个 元 素 ， 根 据 函 数 返 回 的 
boolean 值 构建 新 的 Lisct。 如 果 测 试 结果 为 true， 就 把 这 个 元 素 添 加 到 输出 List 中 。 
为 了 使 用 过 让 器 ,还 需要 一 个 函数 来 判断 是 否 应 该 把 某 个 元 素 包 括 在 内 。 你 可 以 把 它 想象 成 
一 个 向 每 个 元 素 提问 问题 的 函数 :“ 我 应 该 允许 你 通过 过 滤器 吗 ? ” 
这 种 函数 叫做 谓词 函数 ( predicate function ) 这 里 有 一 个 用 伪 代 码 ( 几乎 就 是 Scala ) 编写 的 
方法 : 
(msg) -> { ‘imsg.get ("status") .equalsIgnoreCase ("CANCELLED") |} 
这 个 函数 接受 一 个 参数 ( msg ) 并 返回 boolean 值 。 如 果 msg 被 取消 了 ， 它 会 返回 false， 
否则 返回 true。 用 在 过 滤器 中 时 ， 它 会 过 滤 掉 所 有 被 取消 的 消息 。 
这 就 是 你 想 要 的 。 在 调用 整合 代码 之 前 ， 你 需要 移 除 所 有 被 取消 的 订单 ,因为 被 取消 的 订单 
不 会 出 现在 派发 数据 库 中 。 
事实 上 ， Java 8 准备 采用 这 种 写法 ( 受到 了 Scala 和 C# 语 法 的 强烈 影响 )。 我 们 在 第 14 章 还 会 
讨论 这 个 主题 ， 但 在 那 之 前 我 们 会 遇 到 几 次 函数 字面 值 ( 也 称 为 lambda 表 达 式 ). 
我 们 接着 往 下 看 ,讨论 一 下 其 他 情况 ， 从 JVM 上 可 用 的 语言 类 型 开始 ( 有 时 候 我 们 也 把 这 称 
为 语言 生态 学 )。 


7.2 语言 生态 学 


编程 语言 有 多 种 不 同 的 流派 和 类 型 。 也 就 是 说 ,不 同 的 语言 有 不 同 的 编码 风格 和 编码 方式 。 
如 果 想 擎 担 这 些 风格 并 让 它们 为 你 所 用 ， 你 得 弄 明 日 这 些 差 异 以 及 如 何 给 声言 分 类 。 


注意 ”这些 分 类 可 以 帮助 思考 语言 的 多 样 性 。 尽 管 某 些 分 法 可 能 更 清晰 ,但 没有 哪 种 是 完美 的 。 


最 近 几 年 ， 语 言 在 添加 新 特性 时 有 路 越 各 种 分 类 的 趋势 。 这 就 是 说 ， 你 最 好 认为 某 种 语言 比 
其 他 语言 “函数 化 程度 更 低 "， 或 者 “虽然 是 动态 类 型 ， 但 必要 时 也 有 可 选 的 静态 类 型 "。 
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我 们 将 要 讨论 的 分 类 是 “解释 型 与 编译 型 "、“ 动 态 与 静态 “命令 式 与 函数 式 ,还 有 在 JVM 
上 重新 实现 的 语言 与 原生 语言 。 通 常 这 些 分 类 用 来 明确 语言 的 边界 , 不 要 用 学 院 化 方式 把 它们 当 
做 完整 精确 的 分 类 。 

Java 是 运行 时 编译 、 静 态 类 型 的 人 令 式 语 言 。 它 强调 安全 性 、 代 人 码 清 晰 、 性 能 ， 并 乐于 表现 
出 一 定 程度 的 柳 开 和 死板 (比如 在 部 站 中 ) 不 同 的 语言 可 能 侧重 不 同 ， 比 如 动态 类 型 的 请 言 可 
能 更 看 重 部 署 速度 。 

我 们 先 从 解释 型 与 编 详 型 分 类 开始 介绍 。 


7.2.1 解释 型 与 编译 型 语言 


解释 型 语言 是 那 种 源码 是 什么 就 执行 什么 的 语言 , 不 会 在 执行 开始 之 前 把 整个 程序 转 成 机 器 
人 码 。 编 译 型 语言 则 和 不同， 一 开始 就 要 用 编译 器 把 人 类 可 读 的 源码 变 成 二 进 制 形式 。 

这 种 分 别 最 近 变 模糊 了 。80 年 代 和 90 年 代 早 期 这 两 类 语言 的 边界 还 相当 清晰 : CC++ 及 类 似 
的 语言 是 编译 型 ，Perl 和 Python 是 解释 型 。 但 Java 同 时 兼 具 编译 型 和 解释 型 两 种 特性 ， 这 一 点 我 
们 已 经 在 第 1 章 讲 过 了 。 字 节 码 的 出 现 使 这 个 界限 更 模糊 了 。 人 类 肯定 读 不 了 字 节 码 ， 但 它 也 不 
是 真正 的 机 器 码 。 

对 于 本 书 中 要 研究 的 JVM 语 言 , 我 们 划分 的 边界 是 该 语言 是 否 会 将 源码 编译 为 类 文件 并 且 执 
行 。 不 产生 类 文件 的 语言 会 由 解释 器 ( 可 能 是 用 Java 写 的 ) 连 行 执行 源码 。 有 些 语言 既 有 编译 器 
也 有 解释 器 ， 还 有 些 既 有 解释 器 又 有 产生 JVM 字 节 码 的 即时 编译 器 (JIT )。 


7.2.2 动态 与 静态 类 型 


在 动态 类 型 语言 中 ， 变 量 在 不 同时 间 可 能 会 有 不 同 的 类 型 。 我 们 以 一 小 段 简单 的 JavaScript 
代码 为 例 , JavaScript 是 著名 的 动态 语言 。 即 便 你 不 了 解 这 种 语言 , 也 应 该 很 容易 理解 下 面 的 代码 : 

下 本 

anawer = "What is the answer? " + AaAnNSwer; 


在 这 段 代码 中 ， 变 量 answer 一 开始 被 赋值 为 40， 当 然 ， 是 个 数值 。 然 后 给 它 加 上 2， 变 成 了 
42。 之 后 我 们 给 answer 赋 了 个 字符 串 值 。 这 在 动态 语言 中 是 非常 普遍 的 技术 ， 不 会 引起 语法 


JavasScript 解 释 器 也 能 分 清 两 种 + 操作 符 的 用 法 。 第 一 个 + 是 数字 相 加 把 2 加 到 40 上 ， 而 在 
下 一 行 中 ,解释 器 能 从 上 下 文中 推导 出 开发 人 员 要 做 字符 串 合并 。 


注意 这 里 的 关键 是 动态 类 型 语言 跟踪 变量 值 的 类 型 ( 比如 数字 或 字符 事 ) 信息 ， 而 静态 类 型 
语言 跟踪 变量 的 类 型 信息 。 


静态 类 型 非常 适合 编译 型 语言 ,因为 所 有 类 型 信息 都 在 变量 上 ，, 跟 变 量 的 值 没有 关系 。 这 样 
很 容易 在 编译 时 推导 潜在 的 类 型 系统 违规 行为 。 
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动态 类 型 语言 把 类 型 信息 放 在 变量 所 持 有 的 值 上 。 也 就 是 说 很 难 提前 发 现 类 型 违规 行为 , 因 
为 推导 所 需 的 信息 直到 运行 时 才能 得 到 。 


7.2.3 ”命令 式 与 函数 式 语 言 


Java 7 是 典型 的 命令 式 语 言 。 命 令 式 语言 把 程序 的 运行 状态 建 模 为 可 修改 的 数据 ， 用 一 系列 
指令 来 改变 运行 状态 。 因 此 ， 在 命令 式 声言 中 ， 程 序 状态 才 是 核心 概念 。 

命令 式 语言 主要 分 为 两 类 。 一 种 是 过 程 语言 ， 比 如 BASIC 和 FORTRAN。 这 种 语言 把 代码 和 
数据 完全 分 开 ， 有 简单 的 代码 操作 数据 范式 。 另 外 一 种 是 面向 对 象 ( 00 ) 语言 ， 数 据 和 代码 ( 以 
方法 的 形式 ) 共同 封装 在 对 象 中 。 面 向 对 象 语言 中 或 多 或 少 地 存在 元 数据 ( 比如 类 信息 ) 引入 的 
额外 结构 。 

函数 式 语言 不 同 , 它 把 计算 本 身 当 做 最 重要 的 概念 。 函 数 式 语言 跟 过 程 语言 一 样 对 值 进 行 操 
作 ， 但 它 不 会 修改 输入 ， 而 是 像 数学 函数 一 样 返回 新 值 。 

如 图 7-2 所 示 ， 晴 数 被 看 做 “小 处 理 机 ”， 输 入 值 并 输出 新 值 。 它 们 没有 任何 自己 的 状态 ,并 
且 把 它们 和 任何 外 部 状态 绑 在 一 起 也 没有 任何 意义 。 这 就 是 说 一 切 箔 对 象 的 世界 观 跟 函 数 式 语言 
的 自然 观点 有 些 分 歧 。 


面向 对 象 方式 


byte data = 


1011 1010 1000; 


国 数 式 方 式 


1011 1010 1000 一 一 一 全 _ 革 。 : 


图 7-2 命令 式 和 函数 式 语言 
在 接 下 来 的 三 章 里 ， 每 章 重 点 介绍 一 种 语言 ， 并 且 都 会 以 前 面 对 隐 数 式 编程 的 讨论 为 基础 。 
我 们 会 从 Groovy 开 始 ， 它 种 “一 点 儿 函 数 式 风格 "， 用 我 们 在 7.1 节 讨论 过 的 方式 处 理 集 合 ; 然后 
是 Scala， 对 FP 的 利用 更 加 充分 ; 最 后 是 Clojure ( 纯粹 的 果 数 式 语 言 ， 完 全 没有 面向 对 象 特性 )。 


7.2.4 ”重新 实现 的 语言 与 原生 语言 


JVMi 语 言 之 间 的 男 一 个 重要 区 别 是 重新 实现 已 有 语言 与 专门 以 JVM 为 目标 的 划分 。 通 常 来 
说 ， 那 些 专门 以 JVM 为 目标 写 的 语言 能 把 自己 的 类 型 系统 跟 JVM 的 原生 类 型 结合 得 更 紧密 。 
下 面 是 三 种 重新 实现 已 有 语言 的 JVYM 语 言 。 


一 一 人 1000 ll1901 1001 
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口 JRuby 在 JVM 上 重新 实现 了 Ruby 语 言 。 Ruby 是 一 个 动态 类 型 的 面向 对 象 语言 , 有 些 函 数 式 
特性 。 它 在 JVM 上 基本 算 解释 型 的 ， 但 最 近 发 布 的 版 本 中 有 一 个 运行 时 JIT 编 译 融 ， 在 适 
当 条 件 下 可 以 生成 JYM 字 节 码 。 

口 Jython 由 Jim Hugunin 在 1997 年 发 起 ， 当 时 是 为 了 在 Python 中 使 用 高 性 能 的 Java 类 库 。 它 在 
JVM 上 重新 实现 了 Python， 因 此 是 动态 的 , 总体 还 算 面向 对 象 语言 。 它 的 运作 方式 是 先 在 
内 部 生成 Python 字 节 三, 然后 再 转换 成 JVM 字 节 码 。 这 使 它 能 在 Python 典型 的 解释 模式 ( 看 
起 来 像 ) 下 工作 。 通 过 生成 JVM 字 节 码 ， 并 把 结果 类 文件 保存 到 硬盘 上 ， 它 也 能 在 预先 
(AOT ) 编译 模式 下 工作 。 

口 Rhino 最 初 是 由 Netscape 开 发 的 ， 后 来 转 绘 Mozilla 项目。 它 在 JVM 上 提供 了 一 个 JavaScript 
实现 。JavaScript 是 动态 类 型 的 面向 对 和 象 语 言 (但 和 Java 实 现 面 向 对 象 的 方式 截然 不 同 )。 
Rhino 既 支持 编译 模式 也 支持 解释 模式 ， 随 Java 7 一 起 发 布 (具体 细节 请 参见 


com.sun.script.javascript 包 ), 


天 二 二 1) i ta i 
3 a ST 全 EE MW 早 的 JVM 语 言 i ne 
下 站 a a We™1 YW 1 二 Ed i 和 ep 
中 由 f | mm = ee % : 1 岗 时 人 1 四 

Shei I | | HF 可 二 人 


了 Tower yp pe 
Kawa 语 言 ， 它 是 一 种 Lisp 语 言 。 在 那 之 后 这 些 语言 几乎 星 现 了 爆炸 式 增长 ， 因 此 违 踪 它 们 的 
历史 太 难 了 。 


在 编写 本 书 时 ， 和 猜测 至 少 有 200 种 JVM 语 言 应 该 是 合理 的 。 不 能 说 所 有 语言 都 很 活 牙 或 得 到 
了 广泛 应 用 ， 但 这 个 数字 起 码 能 表明 JVM 是 一 个 非常 活 婚 的 语言 开发 和 实现 平台 。 


注意 “在 随 Java 7 推出 的 语言 和 VM 规 范 里 ， 所 有 对 Java 语 言 的 直接 引用 都 从 VM 规 范 中 去 掉 了 。 
Java 现 在 只 是 运行 在 JVM 上 的 众多 语言 中 的 普通 一 员 ， 它 不 再 享有 特权 了 。 


就 像 我 们 在 第 5 章 中 讨论 的 , 能 让 这 么 多 不 同 的 语言 运行 在 JVM 上 的 关键 技术 是 类 文件 格式 。 
任何 能 产生 类 文件 的 语言 都 可 以 认为 是 JVM 上 的 编译 型 语言 。 

我 们 接 下 来 会 讨论 多 语言 编程 怎么 变 成 了 让 Java 程 序 员 感 兴趣 的 领域 。 我 们 会 解释 基本 要 
念 ， 为 什么 要 给 我 们 的 项 目 选择 一 种 备 选 的 JVM 语 言 以 及 如 何 操作 。 


7.3 JVM 上 的 多 语言 编程 


“JVM 上 的 多 语言 编程 ”这 种 说 法 还 挺 新 颖 的 。 这 种 说 法 是 为 了 描述 那些 以 Java 代 码 为 核心 ， 
但 还 用 了 一 种 或 多 种 其 他 非 JavaJVM 语 言 的 项 目 。 多 语言 编程 通常 是 一 种 关注 点 分 离 的 形式 。 如 
图 7-3 所 示 ， 非 Java 技 术 的 作用 可 以 分 为 三 个 层次 。 这 张 图 有 时 被 称 为 多 语言 编程 金字 塔 ,这 要 归 
功 于 Ola Bini。 

金字 塔 中 有 三 个 明确 的 层次 : 特定 领域 层 、 动 态 层 和 稳定 层 。 


内 目 在 读书 @ 


WWW.ZiZzidiary.com 
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”多 语言 编程 之 所 以 有 意义 ,是 因为 不 同 的 代码 片段 有 不 同 的 生存 期 。 银行 里 的 风险 引 
能 会 持续 运行 五 年 以 上 ; 而 网 站 上 的 JSP 页 面 可 能 只 有 几 个 月 ; 最 短命 的 启动 代码 可 能 
几 天 。 代 码 “ 活 ”得 时 间 越 长 ， 越 靠近 金字 塔 的 底部 。 上 
它 代表 了 不 同 侧重 点 的 相互 折 中 ， 比 如 底部 更 关注 性 能 和 全 面 测试 ,而 顶部 侧重 的 是 灵活 
性 和 快速 部 署 能 力 。 yr | | a ey 


表 7-1 给 出 了 这 三 个 层次 的 更 多 细节 。 
表 7-1 三 层 备 语言 编程 金字 塔 
名 称 描 述 | ” 例 子 
特定 领域 层 特定 领域 语言 。 与 应 用 程序 领域 的 特定 部 分 结合 Apache Camel DSL、Drools、Web 模 板 
非 汕 紧密 
动态 层 开发 速度 快 ， 生 产 率 高 ， 功 能 灵活 部 署 Groovy, Jython. Clojure 
稳定 层 核心 功能 、 稳 定 、 经 过 良好 袜 试 、 性 能 高 Java. Scala 


这 些 层次 中 有 特定 的 模式 ， 静 态 类 型 硬 言 更 倾向 于 稳定 层 的 任务 。 相 反 ， 能力 不 是 邦 么 强 ，、 
通用 性 比较 低 的 技术 在 金字 塔 的 顶部 更 容易 找到 目 己 的 位 置 。 

金字 塔 中 部 给 动态 语言 留 出 了 很 多 位 置 。 这 也 是 最 灵活 的 一 层 , 大 多 数 情况 下 在 动态 层 内 部 
或 者 在 动态 层 和 相 邻 层 之 间 有 重奏 。 

我 们 要 对 这 张 图 继续 深 控 ， 看 看 Java 语 言 为 什么 不 是 金字 塔 所 有 层次 的 最 佳 选择 。 接 下 来 我 
们 先 来 看 看 为 什么 要 考虑 非 Java 语 言 ， 然 后 给 出 一 些 选择 非 Java 语 言 的 重要 标准 。 


7.3.1 为 什么 要 用 非 Java 语言 


Java 作 为 一 种 通用 、 静 态 类 型 的 编译 型 语言 有 很 多 优势 。 这 些 品 质 使 它 成 为 实现 稳定 层 功能 
的 绝 佳 选 择 。 但 同样 的 特性 放 到 金字 塔 上 层 就 会 变 成 负担 。 比 如 说 ， 

口 重新 编译 太 费 工 了 上; 

口 静态 类 型 不 够 灵活 ， 重 构 起 来 时 间 可 能 会 比较 长 ; 
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口 部 署 的 动静 太 大 ; 
口 Java 的 语法 天 然 不 适用 于 生产 DSL。 
Java 项 目 重新 编译 的 时 长 迅速 攀升 到 了 90 秒 到 ?2 分钟。 这 个 长 度 足 以 严重 打 断 开发 人 员 的 思 
路 ， 并 且 对 于 只 在 生产 环境 中 存活 几 个 星期 的 代码 来 说 ， 这 种 开发 方式 太 糟 粒 了 。 
比较 务实 的 办 法 是 利用 Java 丰 语 的 API 和 类 库 完 成 稳定 层 的 繁重 工作 。 
注意 ”如果 你 刚 开 始 一 个 新 项 目 ， 可 能 会 发 现 其 他 稳定 层 语言 ( 比如 Scala ) 也 具备 非常 重要 的 
特性 ( 比如 单 越 的 并 发 支持 )。 然 而 在 大 多 数 情况 下 ,不 应 该 用 其 他 种 类 的 稳定 语言 重 写 
正在 使 用 的 稳定 层 代 码 。 


这 时 你 可 能 会 纳闷 :“ 每 一 层 都 会 面临 什么 样 的 编程 挑战 ， 我 该 选 娜 种 语言 ? ”一 个 优秀 的 
Java 开 发 人 员 知 道 ， 根 本 没有 所 谓 的 银 弹 ， 但 当 我 们 面 对 选 择 时 ， 的 确 有 一 定 的 评估 标准 。 我 们 
不 可 能 在 这 里 把 每 个 可 能 的 选项 都 讨论 到 ， 所 以 在 剩 下 的 章节 中 ， 我 们 会 集中 讨论 三 种 大 多 数 
Java 开 发 人 员 可 能 都 会 面临 的 选择 。 


7.3.2 新 露头 角 的 语言 新 星 


接 下 来 我 们 会 挑 三 种 , 在 我 们 看 来 可 能 最 有 生命 力 和 影响 力 的 语言 。 这 些 JVM 语 言 ( Groovy、 
Scala 和 和 Clojure ) 在 多 诸 言 程序 员 心 目 中 已 经 有 了 相当 的 分 量 。 这 三 种 语言 为 什么 会 得 到 大 家 的 青 
睐 ? 且 听 我 们 一 一 道 来 。 

1. Groovy 

Groovy 语 言 是 James Strachan 在 2003 年 发 明 的 。 它 是 动态 的 编译 语言 , 语法 跟 Java 很 像 ， 但 更 
灵活 。 它 被 三 涂 用 做 脚本 语言 和 快速 原型 语言 ， 并 且 经 常 是 开发 人 员 或 团队 首选 的 非 Javai 语 言 调 
研 对 象 。 你 可 以 把 Groovy 看 做 是 动态 层 的 语言 , 它 以 擅长 构建 DSL 著 称 。 第 8 章 主 要 介绍 Groovy。 

2. Scala 

Scala 是 一 门面 向 对 象 的 语言 ,但 也 支持 函 数 式 编程 , 它 的 起 源 可 以 追溯 到 2003 年 , 当时 Martin 
Odersky 下 在 用 Java 做 一 个 与 译 型 相关 的 项 目 , 结果 却 催生 了 Scala。 它 和 Java 一 样 , 是 静态 类 型 的 
编译 语言 , 但 和 Java 不 同 , 它 做 了 大 量 的 类 型 推断 工作 。 也 就 是 说 它 经 常 给 人 以 动态 语言 的 感觉 。 

Scala 从 Java 中 借鉴 了 很 多 东西 ， 并 且 它 的 设计 “修正 ”了 Java 中 几 个 长 期 以 来 困扰 开发 人 员 
的 问题 。Scala 可 以 做 稳定 层 语言 ， 并 且 有 些 开发 人 员 认 为 它 可 能 会 取代 Java 成 为 “JVM 上 的 下 一 
个 大 语言 "。 第 9 章 介 绍 Scala。 

3. Clojure 

Clojure 征 由 Rich Hickey 议 计 的 ， 属 于 Lisp 家 族 的 语言 。 它 从 Lisp 中 继承 了 很 多 语法 特性 ( 包 
括 大 量 的 括号 )。 就 像 大 多 数 Lisp 语 言 一 样 ， 它 是 动态 类 型 的 函数 式 语言 。 它 是 编译 型 语言 ， 但 


WD Lisp 的 表达 式 是 一 个 原子 (atom ) 或 表 (list ): 原子 ( atom ) 又 包含 符号 (symbol ) 与 数值 ( number );， 表 是 由 零 
个 或 多 个 表达 式 组 成 的 序列 表达 式 之 间 用 空格 分 陋 ， 放 人 一 对 括号 中 。 此 处 的 括号 应 该 是 指 内 置 的 表达 式 。 
一 详 者 注 
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通常 以 源码 形态 发 布 ( 稍 后 解释 )。Clojure 还 向 它 的 Lisp 核 心中 添加 了 相当 可 观 的 新 特性 ( 特别 


是 在 并 发 方面 ln 


Lisp 通 常 被 当做 专家 语言 。Clojure 在 某 种 程度 上 来 说 要 比 其 他 Lisp 语 言 容易 掌握 ， 然 而 这 并 
不 会 影响 其 强大 的 力量 ( 也 非常 适合 测试 驱动 的 开发 风格 )。 但 它 可 能 还 是 徘徊 在 主流 之 外 ， 只 
是 狂热 的 爱好 者 手中 的 秘密 武器 ， 抑 或 遇 到 适合 它 的 特殊 工作 才 发 光 ( 比如 有 些 金融 应 用 程序 发 


现 它 的 功能 组 合 非常 有 吸引 力 )。 


Clojure 通 常 被 认为 是 动态 层 的 语言 , 但 由 于 它 的 并 发 支持 以 及 其 他 一 些 特性 , 也 能 胜任 很 多 
稳定 层 语言 的 工作 。 第 10 章 会 重点 介绍 Clojure。 
现在 我 们 已 经 把 可 选择 的 一 部 分 语言 罗列 出 来 了 , 接 下 来 该 讨论 一 下 决定 你 做 出 选择 的 那些 


因素 了 。 


7.4 如 何 挑选 称心 的 非 Java 语言 


一 日 决定 在 项 目 中 实验 非 Java 语 言 , 就 要 先 把 项 目 中 的 各 个 工作 域 分 清楚 : 哪些 属于 稳定 层 、 
哪些 属于 动态 层 或 特定 领域 层 。 表 7-2 中 给 出 了 分 属 各 层 的 工作 。 


名 称 
特定 领域 忆 


动态 层 


适合 稳定 层 、 动 态 层 或 特定 领域 层 的 项 目 域 


说 明 
构建 、 持 续集 成 、 持 续 部 团 
开发 操作 
企业 集成 模式 建 模 
业务 规则 建 模 
快速 Web 开 发 
原型 
交互 式 管 理 与 用 户 控 制 台 
脚本 
测试 (比如 用 十 测试 驱动 或 行为 驱动 的 开发 ) 
并 发 代码 
应 用 容器 
核心 业务 功能 


如 你 所 见 ， 这 些 备 选 语言 的 使 用 范围 非 第 广泛 。 但 确定 用 备 选 语言 解决 哪 项 工作 只 是 开始 ， 
接 下 来 还 要 评估 用 备 选 语言 是 否 合适 。 下 面 是 帮 有 我 们 选择 技术 的 一 些 标准 。 


口 是 否 为 项 目 里 的 低 风 险 区 。 
口 备 选 语言 跟 Java 的 交互 操作 是 否 容 易 。 


口 备 选 语言 是 否 有 工具 支持 ( 如 IDE 支 持 )。 


口 坪 言 学 习 难 度 。 
口 招聘 这 门 语言 的 开发 人 员 的 难度 。 
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我 们 会 逐一 深入 探讨 这 些 标准 。 对 于 自己 应 该 回答 的 问题 ， 你 得 做 到 心中 有 数 。 


7.4.1 和 低 风险 


假设 你 有 一 个 核心 的 支付 处 理 规则 引擎 ,每 天 要 处 理 一 百 万 笔 交 易 。 这 是 一 个 大 概 运 行 了 七 
年 、 稳 定 的 Java 软 件 ， 但 并 没 做 过 太 多 测试 ， 代 码 里 有 大 量 死角 。 对 于 引信 新 语言 来 说 ， 支 付 处 
理 引 擎 显然 是 高 爷 区 ， 更 不 用 说 它 本 来 跑 得 好 好 的 ， 而 且 测 试 没 做 到 全 面 敌 羔 ， 开发 人 员 也 还 没 
完全 弄 明 日 它 是 怎么 回 事 儿 。 

但 系统 不 可 能 只 有 核心 部 分 。 比 如 更 完善 的 测试 明显 会 对 系统 有 帮助 Scala 有 一 个 非常 好 的 
测试 框 课 : ScalaTest ( 我 们 会 在 第 11 章 介绍 )， 可 以 用 来 测试 Java 或 Scala 代 码 。 开 发 人 员 能 用 它 
写 出 跟 JUnit 相 似 但 简洁 得 多 的 测试 代码 。 所 以 一 旦 度 过 了 ScalaTest 的 学 习 曲 线 , 开发 人 员 就 能 非 
党 高 效 地 增加 测试 覆盖 面 。 而 且 ScalaTest 对 于 逐步 在 代码 库 中 引 人 和 人 行为 驱动 开发 这 样 的 概念 很 有 
办 法 。 在 将 来 要 对 核心 的 某 些 部 分 进行 重 构 或 替换 时 ,不管 最 终 新 的 处 理 引 擎 是 用 Java 还 是 用 
Scala 写 ， 能 用 上 现代 测试 特性 真 的 很 有 帮助 。 

或 者 假设 你 需要 建 一 个 Web 控 制 台 ,以便 操作 员 能 管理 支付 处 理 系 统 后 台 一 些 不 太 重 要 的 静 
态 数 据 。 开 发 人 员 都 知道 Struts 和 JSF， 可 对 这 两 种 技术 都 提 不 起 兴趣 。 这 是 男 外 一 个 试用 新 语言 
和 技术 栈 的 低 风险 区 。Grails 是 个 很 抢眼 的 选择 ( 基于 Groovy 的 Web 框 架 ， 受 Ruby on Rails 启 发 )。 
开发 人 员 在 经 过 一 些 研究 后 ( Matt Raible 也 做 过 一 个 非常 有 趣 的 调研 )， 一 致 认为 Grails 是 生产 率 
最 商 的 Web 框 架 。 

因为 是 集中 在 低 风 险 区 的 有 限 试点 上 做 实验 ， 如 果 所 营 试 的 技术 栈 不 适合 自己 的 团队 或 系 
统 ， 经 理 可 以 随时 终止 项 目 ， 不 用 中 断 太 久 就 可 以 转移 到 不 同 的 交付 技术 上 ，。 


7.4.2 与 Java 的 交互 操作 


你 肯定 不 想 把 原来 写 的 那些 Java 代 码 弃 之 不 用 ! 很 多 组 织 都 是 因为 这 个 原因 退 迟 不 肯 引 入 新 
的 编程 语言 。 但 因为 备 选 语言 是 跑 在 JVM 上 的 ， 所 以 可 以 充分 发 挥 原 有 代码 的 作用 ， 这样 问题 变 
成 了 怎么 让 已 有 代码 库 的 价值 最 大 化 ， 而 不 是 抛弃 正在 使 用 的 代码 ，。 

JVM 上 的 备 选 语言 跟 Java 之 间 的 互 操作 简单 利落 ， 当 然 也 能 部 署 到 原先 的 环境 中 。 在 讨论 这 
个 问题 时 一 定 要 请 管理 生产 环境 的 同仁 到 场 参与 。 在 把 非 JavaJVM 语 言 加 入 到 系统 中 时 ， 你 需要 
充分 运用 他 们 的 专业 经 验 。 这 也 有 助 于 消除 他 们 对 支持 新 方案 的 担忧 ， 还 能 降低 风险 。 


注意 ”DSL 一 般 都 是 用 动态 层 语言 构建 的 ( 某 些 情况 下 也 有 稳定 层 语言 )， 所 以 它们 大 多 数 是 通 

过 其 内 置 语言 运行 在 JVM 上 ， 

有 些 语言 跟 Java 交 互 操作 起 来 更 容易 。 我 们 发 现 最 流行 的 JVM 备 选 语言 ( 比如 Groovyy、Scala、 

Clojure、Jython 和 JRuby ) 都 跟 Java 互 操作 得 很 好 ( 而 且 其 中 某 些 语 言 的 集成 做 得 非常 棒 ， 几 乎 天 

衣 无 锋 )， 如 果 你 确实 是 个 谨 小 慎 微 的 人 ， 可 以 先 做 几 个 实验 ,很 快 ， 也 很 容易 ， 而 且 你 肯定 能 
明白 集成 是 如 何 工作 的 。 
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我 们 以 Groovy 为 例 。 在 Groovy 代 码 里 可 以 用 我 们 熟悉 的 import 语 名 直接 导 人 Java 包 。 用 基 
于 Groovy 的 Grails 框 架 可 以 快速 搭建 一 个 网 站 ,而 引用 对 象 仍然 是 Java 模 型 对 象 , 反 过 来 说 ,在 Java 
代码 里 调用 Groovy 代 码 也 非常 容易 , 有 各 种 各 样 的 办 法 , 得 到 的 也 是 熟悉 的 Java 对 象 。 比如 从 Java 
里 调用 Groovy 代 码 处 理 JSON 数 据 ， 并 让 它 返 回 一 个 Java 对 象 。 


7.4.3 良好 的 工具 和 测试 支持 


大 多 数 开 发 人 员 一 旦 习惯 了 已 有 环境 ， 就 会 低估 他 们 能 节省 的 时 间 。 有 强大 的 IDE 和 构建 工 
具 、 测 试 工具 帮 他 们 快速 生产 出 高 质量 的 软件 。Java 开 发 人 员 多 年 来 受益 于 优秀 的 支持 工具 ， 所 
以 一 定 要 记得 其 他 语言 的 成 熟 度 可 能 还 设 达 到 相同 的 水 平 。 

有 些 语 言 ( 比如 Groovy ) 在 编译 、 测 试 和 最 终结 果 部 署 方 面 长 期 以 来 都 有 IDE 的 支持 。 而 其 
他 语言 的 工具 可 能 羽翼 未 丰 。 比 如 说 Scala 的 IDE 就 不 像 Java 的 那么 好 用 ， 但 Scala 粉 觉得 它 的 强大 
和 简洁 完全 可 以 弥补 当前 IDE 的 不 是 。 

还 有 个 相关 的 问题 ， 当 备 选 语言 社区 开发 出 供 自 己 使 用 的 强大 工具 ( 比如 Clojure 的 构建 工具 
Leiningen ) 后 ,可 能 不 太 好 调整 它 去 处 理 其 他 语言 。 这 就 是 说 开发 团队 要 认真 考虑 该 如 何 分 配 项 
目 ， 特 别 是 在 部 署 各 自 独 立 但 又 相互 关联 的 组 件 时 。 


7.4.4 备 选 语言 学 习 难 度 


学 一 门 新 语言 总 归 需 要 时 间 ， 而 且 如 果 开 发 团队 不 熟悉 该 语言 的 范式 ， 时 间 会 更 长 。 如 果 新 
语言 是 面向 对 象 的 ， 并 有 类 C 的 语法 ( 比如 Groovy )， 那 大 多 数 Java 开 发 团队 都 能 轻松 掌握 。 

林 如 果 偏 离 了 这 一 范式 ,偏离 程度 越 大 ，Java 开 发 人 员 觉 得 越 难 学 。Scala 试 图 在 面向 对 人 象 和 
函数 式 两 个 世界 之 间架 起 一 座 桥 梁 , 但 这 种 融合 对 于 大 规模 软件 项 目 是 否 可 行 仍然 没有 定论 。 在 
特别 流行 的 几 种 备 选 语言 中 ，Clojure 可 能 会 带 来 不 可 思议 的 好 处 , 但 开发 团队 在 学 习 Clojure 的 也 
数 式 属 性 和 Lisp 语 法 时 ， 也 需要 非常 多 的 再 培训 工作 。 

还 有 一 种 选择 是 看 看 重新 实现 已 有 语言 的 JVM 语 言 。Ruby 和 Python 都 是 非常 成 熟 的 语言 , 有 
大 量 的 材料 可 供 开 发 人 员 学 习 。 这 些 语言 的 JVM 替 身 对 于 想 末 用 易学 的 非 Javai 语 言 的 开发 团队 来 
说 ， 是 不 错 的 起 点 。 


7.4.5 ”使 用 备 选 语言 的 开发 者 


组 织 必须 考虑 现实 情况 ， 他 们 不 可 能 总 能 筷 到 前 2% 的 人 ( 不 管 他 们 在 广告 里 怎么 忽悠 )， 而 
且 开 发 团队 的 成 员 也 不 会 整 年 一 成 不 变 。 某 些 语言 ， 比 如 Groovy 和 Scala， 已 经 足够 成 熟 了 ， 所 
以 有 相当 的 开发 人 员 可 以 招募 。 但 像 Clojure 这 样 还 在 想 办 法 推广 自己 的 语言 ， 要 找到 优秀 的 
Clojure 开 发 人 员 仍 然 不 太 容 易 。 


警告 ”对 重新 实现 的 语言 的 警告 ; 比如 说 , 很 多 现 有 的 用 Ruby 写 的 包 和 应 用 程序 ,只 在 原始 的 基 
于 C 的 实现 下 测试 过 。 这 就 是 说 要 在 JVM 上 使 用 它们 可 能 会 有 问题 。 在 选择 平台 时 ， 如 果 
你 计划 采用 整个 用 重新 实现 的 语言 编写 的 技术 栈 ， 那 就 应 该 把 额外 的 测试 时 间 考 虑 在 内 。 
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重新 实现 的 语言 (JRuby、Jython 等 ) 在 这 个 问题 上 很 可 能 再 一 次 发 挥 作用 。 简 历 上 有 JRuby 
的 开发 人 员 可 能 很 少 , 但 因为 它 就 是 JVM 上 的 Ruby, 而 Ruby 开 发 人 员 非 常 多 ( 熟悉 C 版 本 的 Ruby 
开发 人 员 很 容易 掌握 在 JVM 上 运行 导致 的 差异 )。 

要 理解 备 选 JVM 语 言 的 设计 选择 及 限制 ， 你 要 先 理解 JVM 如 何 支 持 多 种 语言 。 


7.5 JVM 对 备 选 语言 的 支持 


一 种 语言 要 在 JVM 上 运行 可 能 有 两 种 方式 : 

口 有 一 个 产生 类 文件 的 编 评 闹 ; 

有 一 个 用 JVM 字 下 但 实现 的 解释 疹 。 

无 论 哪 种 方式 ， 通 向 都 会 有 个 运行 时 环境 为 执行 该 语言 编写 的 程序 提供 支持 。 图 7-4 展 示 了 
Java 和 一 种 典型 非 Java 语 言 的 运行 时 环境 栈 。 


Java JVM 语 言 义 


用 户 代 码 用 户 代码 (X) 
CLASSPATH 依 赖 项 语言 X 的 类 库 


语言 X 运 行 时 


va 的 CLASSPATH 依 环 项 


JVM 


图 7-4 “ 非 Java 语 言 运行 时 支持 


这 些 运行 时 支持 系统 复杂 度 各 不 相同 ,主要 取决 于 给 定 的 非 Java 语 言 运行 时 所 要 掌握 的 资源 
数量 。 几 乎 在 所 有 情况 下 ， 运 行 时 都 会 作为 可 执行 程序 类 路 径 (classpath ) 上 的 一 组 JAR 文 件 ， 
并 且 要 在 程序 执行 开始 之 前 启动 。 

本 书 的 重点 是 编译 型 语言 。 至 于 Rhino 等 解释 型 语言 ， 只 是 为 了 内 容 的 完整 性 才 提 - 一 下 ， 所 
以 我 们 不 会 在 上 面 花 太 多 篇 幅 。 在 本 节 剩 下 的 内 容 中 ,我 们 会 介绍 备 选 语言 所 需 的 运行 时 支持 ( 其 
至 还 包括 编译 型 语言 )， 然 后 探讨 编译 器 小 说 ( compiler fiction ) 一 那些 可 能 不 会 出 现在 底层 字 
节 码 中 、 由 编译 器 合成 的 该 语言 特有 的 功能 。 


7.5.1 非 Java 语言 的 运行 时 环境 


有 一 种 评估 语言 运行 时 环境 复杂 度 的 简单 办 法 : 看 运行 时 实现 中 JAR 文 件 的 大 小 。 按 这 个 标 
准 ，Clajure 的 运行 时 环境 量 级 很 既 ， 而 JRuby 语 言 则 需要 更 多 支持 。 

这 个 标准 其 实 不 是 特别 公平 , 因为 有 些 语言 把 很 多 类 库 和 功能 都 打包 在 标准 发 布 版 里 了 , 而 
有 些 却 不 这 么 做 。 但 如 果 不 细 究 的 话 ， 它 是 个 挺 有 用 的 经 验 。 

通常 来 说 ， 运 行 时 环境 是 要 帮助 非 Java 语 言 的 类 型 系统 和 其 他 特性 符合 期 望 的 语义 。 备 选 语 
言 对 基本 编程 概念 的 看 法 有 时 和 Java 并 不 完全 一 致 。 
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比如 ，Java 的 面向 对 象 方式 并 不 是 放 之 四 海 而 篆 准 的 。 在 Ruby 中 ,运行 时 可 以 往 一 个 单独 的 
对 象 里 添加 一 个 额外 的 方法 ,而 这 个 方法 在 定义 类 时 还 不 知道 是 什么 , 也 不 是 在 相同 类 的 其 他 实 
例 上 定义 的 。JRuby 的 实现 需要 把 这 个 属性 ( 不 太 明 白 为 什么 叫 “ 开 放 类 ”) 照搬 过 来 。 而 这 种 高 
级 支持 只 能 放 在 JRuby 的 运行 时 中 。 


7.5.2 ”编译 器 小 说 


十 言 的 某 些 特性 是 由 编程 环境 和 高 层 语 言 台 成 的 , 在 底层 JVM 中 根本 不 存在 。 这些 特 性 称 为 
编译 器 小 说 。 我 们 在 第 6 草 已 经 过 到 过 一 个 例子 了 一 一 Java 的 字符 串 合并 。 


提示 “你 应 该 了 解 一 下 这 些 特性 是 如 何 实现 的 ， 否 则 你 的 代码 可 能 跑 得 慢 ， 基 至 可 能 鲁 掉 整个 
过 程 。 有 时 运行 时 环境 要 做 大 量 的 工作 来 合成 某 个 特性 。 


Java 中 的 编译 硕 小 说 还 包括 检查 型 异 般 和 内 部 类 ( 如 有 必要 ， 内 部 类 总 会 被 转换 成 带 有 特殊 
合成 访问 方法 的 顶层 类 ， 如 图 7-5 所 示 )。 如 果 你 曾经 探究 过 JAR 文 件 的 内 部 ( 用 jar tvf )， 见 到 
过 许多 名 宇 中 有 5$ 的 类 ， 它 们 就 是 被 取出 并 转换 成 “常规 ”类 的 内 部 类 。 


10101101 


图 7-5 用 编译 器 小 说 实现 的 内 部 类 


备 选 语 诗 也 有 编译 带 小 说 。 某 些 情况 下 ， 这 些 编译 右 小 说 甚至 形成 了 语言 的 核心 功能 。 我们 
来 看 两 个 重要 的 例子 。 

1. 函数 是 一 等 什 

我 们 在 7.1 广 介绍 了 函数 式 编 程 的 关键 概念 一 一 函数 应 该 是 能 放 到 变量 中 的 值 。 这 通常 说 成 
“函数 是 一 等 值 "。 我 们 也 指出 Java 对 函数 的 建 模 方 式 并 不 太 好 。 

本 书 第 三 部 分 讨论 的 所 有 非 Java 语 言 都 把 函数 当做 一 等 值 。 也 就 是 说 函数 可 以 放 在 变量 中 、 
传 给 方法 ,并 可 以 像 操作 其 他 任何 值 一 样 操作 。JVM 只 能 把 类 当做 最 小 的 代码 和 功能 单元 ， 所 以 
现在 有 所 有 的 非 Java 讲 言 都 用 小 型 匿名 类 作为 函数 的 载体 ( 不 过 这 在 Java 8 中 可 能 有 所 改变 )。 

解决 源码 和 JVM 字 节 公 之 间 这 种 差异 的 办 法 是 , 记 住 对 象 只 是 把 数据 和 操作 数据 的 方法 绑 在 
一 起 的 东西 。 请 想象 一 个 没有 状态 、 只 有 一 个 方法 的 对 象 ， 比 如 第 4 章 callable 接 口 的 匿名 实现 
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类 。 把 这 样 一 个 对 象 放 到 变量 中 ， 作 为 参数 传递 ， 然 后 调用 它 的 cal1 () 方 法 ,这 一 切 都 很 正常 ， 
像 这 样 ; 
Callable<String> myFn = new Callable<Sstring>() | 
BOVverride 
public String call() 1 
return "The reasult"; 
| 
是 
try 1 
Svystem.out .println{myFn.calll(l})}):;: 
} catceh (Exception e) I 


我 们 把 异常 处 理 忽 略 挥 了 ， 因 为 在 这 个 例子 中 myFn 的 call1) 方 法 不 可 能 抛 出 异常 。 


注意 ”在 这 个 例子 中 ,myFn 变 量 是 一 个 匿名 类 型 ,所 以 在 编译 后 它 看 起 来 应 该 是 个 类 似 NameOf- 
Enclosingclasssl class 的 东西 。 类 的 序号 从 1 开始 ， 并 且 编 译 回 每 遇 到 一 个 就 加 1 。 
如 果 它 们 是 动态 创建 的 ， 并 且 数 量 很 多 (就 像 在 JRuby 语 言 中 那样 )， 会 对 存放 类 定义 的 
PermGen 内 存 区 域 造 成 压力 。 


尽管 Java 对 此 没有 任何 特殊 的 语法 ， 但 Java 程 序 员 经 常用 这 个 技巧 创建 匿名 实现 类 。 这 就 是 
说 结果 可 能 会 有 点 元 长 。 我 们 所 讨论 的 所 有 语言 都 提供 了 编写 这 些 函 数值 (也 叫 函 数字 面值 、 匿 
名 函数 ) 的 特殊 语法 。 它 们 是 函数 式 编 程 风格 的 支柱 ，Scala 和 Clojure 做 得 都 不 错 。 

2. 多 继承 

还 有 一 个 例子 ， 在 Java ( 和 JVM ) 中 没 办 法 表示 实现 的 多 继承 。 实 现 多 继承 的 唯一 办 法 就 是 
使 用 接口 ， 可 它 不 允许 有 任何 具体 的 方法 。 

相反 ,在 Scala 中 特性 机 制 (trait ) 允许 把 方法 的 实现 混合 到 类 中 ， 所 以 它 提供 了 不 同 的 继承 
视图 。 我 们 会 在 第 9 竟 全 面 介 绍 。 现 在 只 要 记 住 这 种 行为 必须 由 Scala 的 编译 器 和 运行 时 合成 ， 在 
VM 层面 不 提供 这 种 特性 。 

对 JVM 上 可 用 的 不 同类 型 的 语言 及 其 独特 功能 的 实现 方式 就 介绍 到 这 里 。 


7.6 小结 


JVM 上 的 备 选 语 言 已 经 有 了 长 足 的 发 展 。 对 于 某 些 特定 的 问题 , 它们 现在 提供 的 解决 方案 要 
比 Java 好 ， 并 且 还 能 与 原来 用 Java 技 术 实 现 的 系统 及 投资 慕容 。 这 就 是 说 ， 即 便 对 于 Java 的 推销 
者 而 言 ，Java 也 不 总 是 所 有 编程 任务 的 首选 。 

了 解 博 言 的 不 同 分 类 方式 ( 静态 类 型 与 动态 类 型 、 命 令 式 与 函数 式 、 编 译 型 与 解释 型 ) 是 为 
不 同 任务 挑选 正确 语言 的 基础 。 
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对 于 多 语言 程序 员 来 说 ， 编 程 语言 大 致 可 以 分 为 三 层 : 稳定 屋 、 动 态 层 和 特定 领域 层 。Java 
和 Scala 这 样 的 语言 最 好 用 来 做 稳定 层 的 软件 开发 ， 而 诸如 Groovy 和 Clojure 等 其 他 语言 更 适合 完 
成 动态 层 或 特定 领域 层 的 任务 。 

某 些 编程 难题 属于 特定 的 层次 ， 比 如 快速 Web 开 发 属于 动态 层 ， 而 建 模 企业 消息 属于 特定 领 
域 层 。 

有 必要 再 次 强调 一 下 , 不 要 在 已 有 生产 系统 的 核心 业务 功能 中 引信 新 博 言 。 对 于 核心 功能 区 
而 言 ， 支 持 级 别 高 、 测 试 覆 盖 率 优异 ， 并 且 有 稳定 的 月 好 记录 非 汕 重要 。 演 其 从 这 里 人 于 ,还 不 
如 选 一 个 风险 低 的 领域 部 署 备 选 语言 。 

不 要 起 了 每 个 团队 和 项 目 都 有 上 自己 独特 的 个 性 , 这 会 影 啊 选择 语言 时 的 决策 。 所 以 这 个 问题 
没有 标准 答案 ,在 选择 一 门 新 语言 时 ,项 目 经 理 和 技术 负责 人 必须 把 项 目 和 团队 的 特性 考虑 在 内 。 

一 个 都 是 经 验 丰 高 的 技术 人 员 组 成 的 小 团队 可 能 会 选择 Clojure， 因为 它 设计 清晰 、 精巧 并 且 
强大 (他们 才 不 管 概念 的 复杂 性 和 招 人 的 难度 呢 )。 而 一 个 Web 团 队 ， 和 希望 团队 能 快速 扩充 ， 能 
吸引 年 轻 人 ， 他 们 可 能 会 因为 生产 率 和 储备 相对 较 丰 富 的 人 才 库 而 选择 Groovy 和 Grails。 

Groovy 、Scala 和 Clojure 是 JVM 语 言 中 的 领头 壮 。 试 完 本 书后 ， 你 能 学 到 这 三 种 最 有 前 途 的 
JVM 备 选 语 言 的 基础 知识 ， 并 让 自己 的 编程 工具 箱 越 来 越 有 意思 。 

下 一 草 我 们 会 学 习 第 一 种 语言 : Groovy。 
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Groovy: Java 的 动 仿 1 


本 章 内 容 

口 为 什么 学 习 Groovy 

口 Groovy 的 基本 语法 

口 Groovy 和 Java 之 间 的 差别 
口 Groovy 之 于 Java 独 有 的 特性 
口 Groovy 为 何 义 是 脚本 语言 
口 Groovy 与 Java 的 互 操作 性 


Groovy 是 一 种 面向 对 象 的 动态 类 型 语言 ， 跟 Java 一 样 运行 在 JVM 上 。 实 际 上 ， 你 可 以 把 它 看 
成 是 给 Java 静 态 世 界 补 充 动态 能 力 的 语言 。Groovy 项 目 最 初 是 由 James Strachan 和 了 Bob McWhirter 4 
在 2003 年 末 创 建 的 ，2004 年 其 领导 者 变 成 了 Guillaume Laforge。 如 今 http://groovy.codehaus.org/ 上 区 
的 Groovy 社 区 仍 在 道 勃 发 展 。 人 们 认为 Groovy 是 继 Java 之 后 最 流行 的 JVMI 河 言 。 

受到 Smalltalk、Ruby 和 Python 的 启发 ，Groovy 已 经 实现 了 几 个 Java 不 具备 的 语言 特 

口 四 数字 面值 ; 

口 对 集合 的 一 等 "支持 ; 

口 对 正则 表达 式 的 一 等 支持 ; 

口 对 XML 处 理 的 一 等 文 持 。 


本， 比如 ， 


注意 ”在 Groovy 中 ， 函 数字 面值 也 叫 闭 包 。 正 如 第 7 章 所 说 的 ， 它 们 且 能 够 放 进 变量 中 的 函数 ， 
能 传递 给 方法 ， 能 像 其 他 值 一 样 操作 。 


那么 我 们 为 什么 要 用 Groovy 呢 ? 如果 你 还 记得 第 7 章 的 多 语言 编程 金字 塔 ， 也 应 该 还 记得 
Java 不 是 解决 动态 层 问题 的 理想 语言 。 这 些 问题 包括 快速 Web 开 发 、 原 型 设计 、 脚 本 人 处理 以 及 其 
他 很 多 问题 。Groovy 就 是 要 解决 这 些 问 题 。 

这 里 有 一 个 体现 Groovy 价 值 的 例子 。 比 如 老板 让 你 写 一 个 Java 例 程 ， 把 一 堆 Java bean 转 成 
XML。 用 Java 当 然 可 以 完成 这 个 任务 ， 和 而 且 有 很 多 API 和 类 库 可 以 选用 ， 


山 这 里 所 说 的 “一 等 ” 基 指 内 置 到 语言 的 辜 法 中 ,不 需要 调用 类 库 。 
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口 Java 6 中 的 Java Architecture for XML Binding ( JAXB ) 和 Java API for XML Processing 
(JAXP ); 

口 Codehaus 上 的 XStream 类 库 ; 

口 Apache 的 XMLBeans 类 库 。 

当然 还 不 止 这 些 ....， 

这 个 过 程 很 费 工 。 比 如 说 ， 要 在 JAXB 下 输出 Person 对 象 对 应 的 XML， 你 必须 写 出 : 


JAXBContext je = JAXBContext .newInstance |"net .teamagparg.domain"),! 
ObjectFactory objFactory = new ObjectFactory(); 

Person person = (IlPerson})objFactory.createPerson'\): 
Dergson.SgetId(2); 

Person.,setName ("Gweneth"):; 

Marshaller m = jc.createMarshaller');} 

m. SetProperty (Marshaller .JAXB FORMATTED OUTFUT, Boolean.TRUE), 
m.marshal (person, System.out).; 


Person 类 必须 是 标准 的 Java bean， 有 完整 的 获取 和 设置 方法 。 
Groovy 的 方式 与 此 不 同 ， 因 为 它 把 XML 当做 一 等 公民 对 待 。 这 是 上 面 那 段 代 码 的 Groovy 实 现 : 


def writer = new StringWriter'(); 
def xml = new groovy .xml .MarkupBuilder (writer).; 
Xml .person{(id:2) 1 
name ‘Gweneth' 
age 1 
| 


println writer.toSstringt{):; 

看 到 了 吧 ， 用 这 种 语言 写 代码 非常 快 ， 并 且 它 和 Java 很 像 ，Java 开 发 人 员 很 容易 掌握 。 

Groovy 还 可 以 帮 你 减少 套路 代码 的 编写 工作 ， 比 如 Groovy 处 理 XML 和 循环 遍历 集合 的 方式 
要 比 Java 更 和 价 洁 。 因 为 Groovy 跟 Java 的 互 捍 作 性 很 好 ， 所 以 在 Java 中 很 容易 利用 Groovy 的 动态 性 
和 语言 特性 。 

因为 跟 Java 的 语法 很 像 ， 所 以 对 于 Java 开 发 人 员 来 说 ，Groovy 的 学 习 曲 线 很 平滑 ， 并 且 只 要 
有 Groovy 的 JAR 就 可 以 开始 了 。 升 望 你 看 完 本 章 后 能 跟 新 语言 拍档 默契 ! 

既然 Groovy 在 由 JVM 执 行 之 前 经 过 了 完整 的 分 析 、 编译 和 生成 过 程 , 有 些 开 发 人 员 会 想 : "为 
什么 它 不 能 在 编译 时 把 那些 明显 的 错误 挑 出 来 呢 ? ”要 记 住 Groovy 是 动态 语言 , 它 的 类 型 检查 和 
绑 定 都 是 在 运行 时 做 的 。 
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i i i oA Ss te ' 半 
求 很 严格 ， wm Groovy 的 对 象 都 扩展 自 
Groovyobject， 它 有 a he (String name, Object args) 方 法 。Groovy 
的 方法 不 能 像 Java 中 那样 直接 调用 ， 而 是 通过 之 前 提 到 的 invokeMethod (string name， 
Object args) 方 法 执行 . 这 个 方法 会 自 Te 和 查找 , 这 自 然 会 降低 处 理 速 度 。 
Groovy 语 言 的 开发 者 已 经 做 了 一 些 优 化 ， 在 接 下 来 的 新 版 本 中 ，Groovy 会 利用 JVM 
的 字 节 码 invokedvnamic 做 更 多 优化 。 
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Groovy 在 做 某 些 重活 时 还 是 靠 Java， 并 且 它 要 调用 已 有 的 Java 类 库 很 容易 。 因 为 Groovy 能 和 
Java 一 起 使 用 它 的 动态 类 型 和 新 语言 特性 ， 所 以 它 是 一 种 卓越 的 快速 原型 设计 语言 。Groovy 还 能 
作为 脚本 处 理 语 言 使 用 ， 因 此 它 以 Java 的 灵活 、 动 态 伴侣 而 著称 。 

本 草 一 开始 会 跑 几 个 简单 的 Groovy 例 子 , 一旦 你 熟悉 了 简单 Groovy 程 序 的 运行 , 就 可 以 开始 
学 习 Groovy 的 具体 语法 了 ,还 有 对 于 Java 开 发 人 员 来 说 比较 难 缠 的 Groovy 内 容 。 本 章 接 下 来 就 深 
挖 一 下 Groovy 的 真 金 真 银 ， 讨论 Java 不 具备 的 几 个 语言 特性 。 最 后 ， 你 可 以 学 习 在 JVM 上 混合 使 
用 Java 和 Groovy 代 码 ， 成 为 一 名 多 语言 程序 员 。 


8.1 Groovy 入 门 


如 采 你 还 没 狐 Groovy， 请 先 参 照 附 录 C 在 你 的 机 费 中 把 它 措 起 来 ， 然 后 再 编译 和 运行 本 章 的 
第 一 个 例子 。 

本 节 会 向 你 展示 如 何 用 命令 行 编 译 和 执行 Groovy， 以 便 你 在 任何 操作 系统 上 都 能 应 用 自如 。 
我 们 还 会 介绍 Groovy 醛 制 台 ， 一 个 宝 贯 的 、 操 作 系统 无 关 的 暂 存 器 环境 ， 非 常 适合 用 来 练 手 。 

装 好 了 吗 ? 那 我 们 就 来 编译 一 些 Groovy 代 码 ， 让 它们 跑 起 来 吧 ! 


8.1.1 编译 和 运行 


这 里 有 些 你 应 该 了 解 的 Groovy 命 令 行 工具 ， 特 别 是 编译 器 ( groovyc ) 和 运行 时 执行 器 
( 2gITOOVY Ds 它们 两 个 基本 上 就 相 当 于 javac 和 java。 
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tee 再 向 你 推荐 一 本 非常 优秀 的 书 : Kenneth A. Kousen 编 著 的 Making 。 
Groovy ( Manning ，2012 )- 


我 们 来 看 一 个 简单 的 Groovy 脚 本 ， 它 可 以 输出 下 面 的 内 容 "， 也 借 此 熟悉 一 下 命令 行 工 具 : 
It'sg Groovy baby, yeah! 
打开 命令 行 提示 竺 ， 执 行 如 下 操作 。 
(1) 随便 找 个 目录 ， 在 里 面 创建 一 个 HelloGroovy.groovy 文 件 。 
(2) 编辑 这 个 文件 ， 加 上 这 一 行 ; 
System.out .printlnl"It's Groovy baby, yeah!"); 
(3) 保存 HelloGroovy.groovy。 
(4) 用 下 面 这 个 命令 编译 它 : 


groovyce HelloGroovy .可 TOOVY 
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(5) 用 下 面 这 个 命令 运行 它 : 
groovy HelloGSroovy 
提示 ”如果 Groovy 源 文件 在 CLASSPATH 下 ， 可 以 跳 过 编译 。 如 果 需 要 ，Groovy 运 行 时 会 先 在 源 
文件 上 执行 groovyc。 


恭喜 ， 你 刚刚 运行 了 有 生 以 来 第 一 行 Groovy 代 三 ! 

跟 Java 一 样 ， 你 可 以 在 命令 行 中 编写 、 编 译 和 执行 Groovy 代 码 , 但 要 处 理 CLASSPATH 之 类 的 
事情 时 ,你 很 快 就 会 觉得 这 么 做 太 征 了 ,主流 的 Java IDE ( Eclipse、IntelliJ 和 NetBeans ) 对 Groovy 
的 支持 都 很 好 , 但 Groovy 也 提供 了 一 个 控制 台 供 你 运行 代码 。 这 个 控制 台 非 常 适合 快速 演练 小 型 
解决 方案 或 原型 ， 因 为 用 它 比 用 正式 的 IDE 快 得 多 。 


8.1.2 ”Groovy 控制 台 


本 章 会 用 Groovy 控 制 台 运行 示例 代码 ， 因 为 它 是 一 个 好 用 、 轻 量 的 IDE。 要 启动 控制 台 ， 请 
在 命令 行 中 执行 groovyConso les 
它 会 弹出 一 个 类 似 图 8-1 这 样 的 独立 窗口 。 
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图 8-1 ” ”Groovy 控制 台 

自 先 ， 你 应 该 取消 勾 选 View ( 视图 ) 菜单 中 的 Show Scriptin Output ( 在 输出 中 显示 脚本 ) 选 

项 。 这 会 让 输出 简单 一 点 儿 。 现在 你 可 以 运行 一 下 前 面 那 个 例子 中 的 Groovy 代 码 ， 以 确保 控制 台 
能 正常 工作 。 在 控制 台 的 项 部 面板 中 输入 下 面 这 行人 代码，; 


System.out .println(l"It's Groovy baby, vyeah!"): 
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然后 点 击 Execute Script ( 执行 脚本 ) 按钮 ， 或 者 用 快捷 名 
面板 中 显示 如 下 输出 : 

It's Groovy baby, yeahl! 

如 你 所 见 ， 输 出 面板 显示 了 刚刚 执行 的 那个 表达 式 的 计算 缮 果 。 

现在 你 已 经 知道 如 何 快速 执行 Groovy 代 码 了 ， 是 时 候 学 一 些 Groovy 的 语法 和 语义 了 。 


8.2 Groovy 101: 语法 和 语义 


Ctrl- 民 。Groovy 控 制 台 就 会 在 底部 


上 一 节 只 写 了 一 行 Groovy 语 句 ， 没 有 任何 类 或 方法 之 类 的 结构 ( 用 Java 时 会 需要 )。 实 际 上 
你 写 的 是 一 pid 


跟 Java 不 同 ，Groovy 的 ; pr 比如 议 ， 因果 体 有 一 和 码 放 在 类 定义 之 
外 pi RR 1 他 动态 脚本 语言 ( 比如 by ES” - 样 ，Groo oovy 肢 本 7 


存 到 .groovy ， 就 可 以 作 为 脚本 运行 。 二 人 和 7 经 用 
jrooy ,和 本 取代 了 shell 肢 本 ， 因为 它们 功能 更 强 ， 更 易于 编写 ， 并且 只 要 装 了 JVM，, 就 可 以 在 
任何 平台 上 运行: 给 你 一 个 性 能 方面 的 小 提示 ， 请 使 用 groovyserv 类 库 ， 和 
Groovy 扩 展 ， 让 脚本 运行 得 更 快 。 | 

Groovy 的 一 个 关键 特性 是 可 以 使 用 跟 Java 中 一 样 的 结构 ,语法 也 类 似 。 为 了 突出 这 种 相似 
性 ， 请 在 Groovy 控 制 台 中 执行 下 面 这 段 类 似 Java 的 代码 ， 


public class PrintSstatement 


] Public static void main(lStringl] args) 
{ 


System.out .printlni"Iit's Groovy baby, yeah!"); 
} 
结果 和 前 面 那 个 只 有 一 行 的 Groovy 脚 本 一 样 ， 都 是 输出 "It's Groovy baby, yeah!"。 
除 丁 使 用 Groovy 控 制 台 ,你 还 可 以 把 源码 放 到 PrintStatement.groovy 源 文件 中 ,用 groovyc 编 译 它 ， 
然后 用 groovy 执 行 。 换 句 话说 ， 你 能 像 Java 中 那样 带 着 类 和 方法 编写 Groovy 源 码 。 


提示 。， 在 Groovy 中 几乎 可 以 使 用 所 有 Java 普 通 语 法 ， 所 以 while/for 循 环 、if/else 站 构 、 
switch 语 句 等 ， 都 会 按 你 期 望 的 方式 工作 。 所 有 新 语法 及 主要 差异 都 会 在 本 节 及 相应 章 
节 中 重点 阐述 。 
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随 着 本 章 内 容 的 深入 ， 我们 会 癌 你 介绍 Groovy 特 有 的 语法 惯用 语 ， 例子 也 会 从 类 似 Java 的 硬 
法 向 更 纯粹 的 Groovy 语 法 转变 。 你 已 经 习 辟 了 结构 沉重 的 Java 代 码 ， 青 见 到 像 脚 本 一 样 简 清 的 
Groovy 硬 法 ， 很 容易 发 现 两 者 的 差异 。 

本 节 的 剩余 部 分 会 介绍 Groovy 的 基本 语法 和 语 羡 ,以 及 它们 为 什么 能 帮助 开发 人 人员。 具体 来 
说 ， 我 们 会 探讨 : 

口 默认 导 人 ; 

口 数字 处 理 ; 

口 变量 、 动 态 与 静态 类 型 ， 以 及 作用 域 ; 

口 列表 与 映射 的 语法 。 

首先 , 理解 Groovy 提 供 了 哪些 开 箱 即 用 的 东西 很 重要 。 我 们 先 来 看 看 Groovy 脚 本 或 程序 的 默 
认 寻 人 。 


8.2.1 默认 导入 


Groovy 会 默认 守信 一 些 语言 包 和 工具 包 , 以 提供 基本 的 语言 支持 。Groovy 还 会 导 人 一 系列 的 
Java 包 ， 以 便 为 其 初始 功能 提供 更 厂 谤 的 基础 。 下 面 这 个 导 人 列表 总 是 隐 售 在 Groovy 代 码 之 中 : 


Daroovy. lang.* 


LD groovy .util.* 

DD java.lang.* 

DD java,io.* 

0 java.math.BigDecimal 
java.math.BigInteger 
LL java.net.* 


java.util,.* 
要 使 用 更 多 的 包 和 类 ， 可 以 像 Java 一 样 用 import 请 句 。 比 如 要 从 Java 中 得 到 所 有 Math 类 ， 
只 要 在 Groovy 源 人 友 里 加 上 jimport java.math.*; 就 行 了 。 


了 添加 功能 (比如 ppg ) ep we Ee z sa " 
为 此 提供 了 一 个 惯用 语 : 通常 是 在 脚本 中 使 用 @Grab 注 解 .另外 一 种 办 法 ( 在 你 仍 在 学 习 Groov 
时 ) 是 效仿 Java， 把 JAR 文 件 加 到 CLASSPATH 中 。 


下 面 就 来 使 用 一 下 默认 的 语言 支持 ， 并 看 看 Java 和 Groovy 在 数字 处 理 上 的 差异 。 


8.2.2 ”数字 处 理 
Groovy 能 动态 计算 数学 表达 式 , 并 且 它 采用 最 小 意外 原则 。 这 一 原则 在 处 理 浮 点 数 时 ( 比如 
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3.2 ) 尤其 明显 。Groovy 在 底层 用 Java 中 的 BigDecimal 表 示 浮 点 数 ， 但 它 会 确保 BigDecimal 的 
行为 尽量 符合 开发 人 员 的 期 望 。 

1. Java 和 BigDecimal 

我 们 来 看 一 个 经 常会 让 开发 人 员 头 疼 的 数字 处 理 问 题 。 在 Java 中 ， 如 果 在 Bigpecimal 3 上 
加 0.2， 你 觉得 答案 应 该 是 什么 ?缺乏 经 验 的 Java 开 发 人 员 在 没 看 Javadoc 的 情况 下 很 可 能 会 执行 
下 面 这 种 代码 , 它 会 返回 一 个 极其 忍 怖 的 结果 : 3.20000000000000001110223024625156540423631 
GOB80908203123。 


BigDecimal Xx = new BigDecimal (3); 
BigDecimal YY = new BigDecimal (0.2}),， 
Svatem.out .println (x.add (ly)); 


经 验 丰富 的 Java 开 发 人 员 知 道 最 好 用 BigDpecimal(String val) ， 而 不 是 用 将 数字 作为 参 
数 的 Bigpecimal 构 造 方法 。 以 字符 串 为 参数 的 构造 方法 写 出 来 的 代码 会 产生 预期 答案 3.2: 
BigDecimal X = new BigDecimal ("3"]); 


BigDecimal YY = new BigDecimal ("0.2"); 
System.out .println(tx.ada(y)):; 


这 有 点 悖 于 常理 ， 所 以 Groovy 默 认 采用 了 以 字符 串 为 参数 的 构造 方法 ， 解 决 了 这 一 问题 。 

2. Groovy 和 BigDecimal 

在 Groovy 中 处 理 浮 点 数 (在 底层 用 BigDecimal 表 示 ) 时 ， 会 自动 使 用 以 字符 串 为 参数 的 构 
造 方法 ，3 + 0.2 会 得 到 3.2。 你 可 以 在 Groovy 控 制 台 中 输入 下 面 的 指令 亲自 证 实 一 下 : 

+ .2)} 

你 会 发 现 Groovy 对 BEDMAS" 的 支持 是 正确 的 。 并且 在 需要 时 能 无 颖 切换 数字 类 型 ( 比如 int 
和 aouble )。 

用 Groovy 进 行 数学 运算 比 Java 简 单 。 如 果 你 想 了 解 展 层 细节 ， 可 以 访问 httpJ/groovycodehaus.org/ 
Groovy+Math， 那 里 有 所 有 的 细节 信息 。 

接 下 来 我 们 学 习 Groovy 如 何 处 理 变量 和 作用 域 . 因为 Groovy 的 动态 性 和 执行 脚本 的 能 力 , 它 
在 这 方面 的 语义 规则 和 Java 稍 有 不 同 。 


8.2.3 变量 、 动 态 与 静态 类 型 、 作 用 域 


因为 Groovy 是 一 种 能 作为 脚本 语言 的 动态 语言 ,所 以 你 要 清楚 动态 类 型 和 静态 类 型 一 些 细微 
差别 ， 还 需要 了 解 Groovy 如 何 限 定 变 量 的 作用 域 。 


提示 ”如果 你 意 在 让 Groovy 代 码 与 Java 互 操作 ， 它 也 能 在 可 能 的 情况 下 使 用 静态 类 型 ， 因 为 它 
简化 了 类 型 重 载 和 调度 机 制 。 


四 回想 起 你 在 学 校 的 日 子 了 吧 ! BEDMAS 表 示 括 号 、 次 方 、 除 法 、 乘 法 、 加 法 和 减法 ， 是 我 们 计算 数学 题目 时 所 要 
再 循 的 顺序 〈 先 计 算 插 号 和 次 方 ， 再 计算 乘除 ， 最 后 计算 加 减 )， 由 于 地 区 不 同 ， 你 的 记忆 中 可 能 是 BODMAS 或 
PEMDAS. 
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首先 你 要 理解 Groovy 动 态 类 型 和 静态 类 型 的 差别 。 

1, 动态 类 型 与 静态 类 型 

Groovy 是 动态 语言 ， 所 以 不 必 指 定 变量 的 类 型 ， 变 量 的 类 型 是 在 声明 (或 返回 ) 时 确定 的 。 
比如 说 ， 你 可 以 把 一 个 Date 赋 值 给 变量 x， 然 后 紧 接 着 青 用 不 同 的 类 型 给 x 赋值 。 

其 三 new Datet{():; 

x = 11} 


用 动态 类 型 能 让 代码 更 简洁 ( 忽略 显而易见 的 类 型 信息 )， 反 馈 更 快 ， 并 且 很 灵活 ， 可 以 在 
一 个 变量 上 赋予 不 同类 型 的 对 象 来 完成 工作 。 对 于 那些 想 对 自己 使 用 的 类 型 更 有 把 握 的 人 ， 

Date x = new Date1l) ; 

如 果 声 明了 静态 类 型 变量 ， 在 用 不 正确 的 类 型 值 对 它 赋 值 时 ，Groovy 能 检查 出 来 。 比 如 : 


Exception thrown 


org .codehause.groovy .runtime.typehandling.GroovyCaatException; Cannot cagst 
object Thu Oct 13 12:58:28 BST 2011°" with class 'jJava.util.Date' to 
class 'double' 


在 Groovy 控 制 台 中 运行 下 面 的 代码 ， 就 可 以 重 现 上 面 的 输出 。 
double vy = -3.14993927 
YY = new Datel(}: 


如 你 所 料 ，Data 类 型 的 值 不 能 赋 给 double 变 量 。Groovy 中 的 动态 和 静态 类 型 都 讨论 到 了 ， 
那 作 用 域 呢 ? 

2. Groovy 中 的 作用 域 

对 于 Groovy 里 的 类 ， 其 作用 域 跟 Java 一 样 ， 类 、 方 法 、 循 环 作用 域 的 变量 ， 它 们 的 作用 域 都 
跟 你 想 的 一 样 。 但 涉及 Groovy 脚 本 时 ， 这 个 话题 就 变 得 比较 有 意思 了 。 


提示 。 记 住 ， 作 为 脚本 的 Groovy 代 码 不 在 平常 的 类 和 方法 结构 中 。8.1.1 节 已 经 给 过 一 个 例子 了 。 


简单 说 ，Groovy 肢 本 有 两 种 作用 域 。 

口 绑 定 域 ， 绑 定 域 是 脚本 的 全 局 作用 域 。 

口 本 地 域 ， 本 地 域 就 是 变量 的 作用 域 局 限于 声明 它们 的 代码 块 。 对 于 在 脚本 代码 块 内 声明 

的 变量 ( 比如 在 脚本 的 顶部 )， 如 果 是 定义 过 的 变量 ， 其 作用 域 就 是 定义 它 的 本 地 域 。 

能 在 脚本 中 使 用 全 局 变量 可 以 极 大 提高 代码 的 灵活 性 。 它 和 Java 中 类 范围 内 的 变量 有 点 像 
定义 变量 是 指 被 声明 为 静态 类 型 ， 或 用 特殊 的 aef 关 键 字 定义 的 变量 (表明 它 是 未 确定 类 型 的 定 
义 蛮 量 )。 

在 脚本 中 声明 的 方法 访问 不 了 本 地 域 。 如 果 你 调用 一 个 试图 引用 本 地 域 中 的 变量 的 方法 , 会 
提示 类 似 下 面 的 错误 消息 : 
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groovy. lang.MissingPropertyException: No such property: hello tor class: 
listing 8 2 

下 面 是 产生 该 异常 的 代码 ， 说 明了 作用 域 的 这 个 问题 。 

string hello sa "Hellol!l": 

void checkHello!'() 


System.out .printlnlhello); 


checkHellol(); 

如 果 用 hello = "Hello!" 换 掉 上 面 代码 里 的 第 一 行 ， 这 个 方法 可 以 成 功 输出 “Hello”"。 因 
为 hello 不 再 定义 为 String， 它 现在 的 作用 域 是 绑 定 域 。 

除了 编写 Groovy 脚 本 时 的 这 些 差 异 , 动态 和 静态 类 型 、 作 用 域 、 变 量 声明 都 跟 你 想 的 完全 一 
样 。 接 下 来 我 们 去 看 看 Groovy 内 置 的 集合 (列表 和 映射 ) 支持 。 


8.2.4 列表 和 了 映射 语法 


Groovy 把 列表 和 映射 (包括 集合 ) 结构 当做 语言 中 的 一 等 公民 对 待 ， 所 以 没 必要 像 Java 奢 样 
显 式 声明 List 和 Map 结 构 。 也 就 是 说 ，Groovy 中 的 列表 和 映射 在 底层 是 由 Java ArrayList 和 
LinkedHashMap 实 现 的 。 

使 用 Groovy 语 法 最 大 的 优势 在 于 可 以 省 掉 很 多 套路 化 的 代码 , 让 代码 更 简洁 , 但 丝毫 不 影响 
可 读 性 。 

Groovy 用 方 括号 [1 指定 和 使 用 列表 结构 ( 是 不 是 想起 了 Java 中 的 原生 数组 语法 )。 下 面 的 代 
码 展 示 了 如 何 引 用 第 一 个 元 素 ( Java )， 获 取 列 表 大 小 (4 )， 以 及 将 列表 设置 为 室 [] 。 

jvmLanguages = ["Java", "Groovy", "Scala", "Clojure"]:; 

println (jvmbLanguages [0]); 

println(jvmLanguages .size()); 


jmLanguages = []; 
println(ljvmLanguages); 


看 ，Groovy 将 列表 作为 一 等 公民 处 理 要 比 用 java .util .List 及 其 实现 类 的 代码 轻 量 得 多 。 

因为 Groovy 是 动态 类 型 语言 ,我们 可 以 把 不 同类 型 的 值 保 存在 列表 ( 或 映射 ) 中 , 所 以 下 面 
的 代码 也 是 正确 的 : 

jmbLanguages = ["Java", 2, "Scala", new Date(})]; 

Groovy 人 处 理 映射 也 跟 这 差不多 ， 用 [] 符 号 ， 并 用 冒号 ( : ) 来 分 开 键 / 值 对 。 以 映射 . 键 的 方 
式 引 用 映射 中 的 值 。 下 面 的 代码 通过 相应 的 操作 展示 了 这 些 功能 ， 

口 引用 键 "Java" 的 值 100; 

口 引用 键 "clojure" 的 值 "N/A"; 

口 将 键 "clojure" 的 值 变 成 75; 

口 将 映射 设 为 空 ( [:] )。 
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languageRatings = [Java:100, Groovy:99, Clojure: "N/A"]; 
printin(languageRatings["Java"])}; 
println(languageRatings.Clojure),; 
languageRatings["Clojure"] = 75; 
println(languageRatings["Clojure"|]); 

languageRatings = |[:]; 

println languageRatings; 


提示 “你 有 没有 注意 到 映射 里 的 键 是 不 带 引 号 的 字符 事 ? 为 了 让 代码 更 简洁 ，Groovy 对 这 个 语 
法 也 做 了 调整 ， 映 射 键 的 引号 可 用 可 不 用 。 


这 种 写法 很 直观 ， 用 起 来 也 舒服 。Groovy 把 对 映射 和 列表 内 置 支持 的 概念 更 进 了 一 步 。 

还 有 一 些 语法 技巧 , 比如 引用 集合 中 一 定 范围 内 的 元 素 , 甚至 可 以 用 负 索 引 引 用 最 后 一 个 元 
素 。 下 面 的 代码 引用 了 列表 中 的 前 三 个 元 素 ( [Java，Groovy，Scala] ) 和 最 后 一 个 元 素 
(Clojure ). 

jvmLanguages = ["Java", "Groovy", "Scala", "Clojure"]:; 

println(jvmbLanguages [0. .2]); 

printlnljvmLanguages[-1]); 

现在 , 我 们 已 经 了 解 了 Groovy 的 一 些 基本 语法 和 语义 。 但 在 真正 使 用 Groovy 之 前 ,还 需要 学 
习 更 多 内 容 。 下 一 节 会 更 深入 地 探讨 Groovy 的 语法 和 语义 , 重点 讲解 Java 开 发 人 员 学 习 Groovy 过 


程 中 那些 “ 难 缠 的 内 容 ”。 


8.3 与 Java 的 差异 一 一 新 手 陷阱 


目前 你 基本 上 已 经 熟练 掌握 Groovy 的 基本 语法 了 ， 当 然 其 中 部 分 原因 是 它 跟 Java 语 法 很 像 。 
但 这 种 相似 有 时 可 能 是 “ 诈 ”， 所 以 本 节 来 讨论 那些 经 营 困 扰 Java 开 发 人 员 的 语法 。 

Groovy 有 大 量 可 以 省 略 的 硬 法 ， 比 如 : 

口 语句 第 束 处 的 分 号 ; 

口 返回 语句 ( return ); 

口 方法 参数 两 边 的 括号 ; 

口 public 访 问 限 定 符 。 

这 类 设计 是 为 了 让 源码 更 简洁 ， 在 快速 设计 原型 时 都 能 体现 出 优势 来 。 

其 他 修改 包括 去 掉 已 检查 和 未 检查 异常 之 间 的 区 别 , 相等 概念 的 替代 办 法 ,以 及 不 再 使 用 内 
部 类 。 我 们 先 从 最 简单 的 开始 :可 选 的 分 号 和 退回 场 句 。 


8.3.1 可 选 的 分 号 和 返回 语句 


在 Groovy 中 ,语句 结束 处 的 分 号 ( ; ) 是 可 选 的 ， 除非 一 行 中 有 多 条 语句 ， 否 则 都 可 以 省 咯 。 
此 外 ， 从 方法 中 返回 对 象 或 值 时 不 必 使 用 return 关 键 字 。Groovy 会 自动 返回 最 后 一 个 表达 
式 的 计算 结果 。 
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代码 清单 8-1 演 示 了 这 些 可 选 语法 ， 并 返回 了 方法 中 最 后 一 个 表达 式 的 计算 结果 3。 
代码 清单 8-1 分 号 和 返回 语句 可 以 省 略 


scratchpad pad = new Scratchpad!l) 
println'lipad.doStuff(})) 


public class Scratchpad 


public Integer doStuff() 


def Xx = 1 
def vy; def String 2 = "Hellor"; | 
汉 玻 可 本 一 一 一 一 一 一 一 一 一 
} 
} 


上 面 的 代码 看 起 来 跟 Java 还 是 很 像 ， 而 Groovy 的 简洁 风格 其 实 还 要 更 强 。 接 下 来 你 会 看 到 
Groovy 如 何 省 略 方法 参数 两 边 的 括号 。 


8.3.2 ”可 选 的 参数 括号 

如 果 Groovy 里 的 方法 调用 至 少 有 一 个 参数 , 并 且 没 有 二 义 性 , 则 可 以 省 略 括号 。 也 就 是 说 下 
面 的 代码 

println("It's Groovy baby, yeah!") 

可 以 写成 

println "It's Groovy baby, yeah!" 

代码 变 得 更 简洁 了 ， 可 读 性 仍然 没 受 影响 。 

下 一 个 特性 是 可 选 的 public 访 问 限定 答 ， 青 用 上 它 ，Groovy 代 码 看 起 来 就 不 太 像 Java 本 。 


没有 返回 语句 


8.3.3 ”访问 限定 符 

优秀 的 Java 开 发 人 员 都 知道 确定 类 、 方 法 和 变量 的 访问 级 别 是 面向 对 象 设计 的 重要 组 成 部 
分 。 跟 Java 一 样 ，Groovy 也 有 public、private 和 protected 级 别 ; 但 和 Java 不 同 ，Groovy 的 
默认 访问 级 别 是 public。 所 以 我 们 把 代码 清单 8-1 改 一 下 ， 去 掉 一 些 默 认 的 public 限 定 符 ， 加 
几 个 privte 限 定 符 ， 如 代码 清单 8-2 所 示 。 
代码 清单 8-2 ”public 是 默认 访问 限定 符 


scratchpad2 pad = new Scratchpad2() 
println(lpad.doStuff ()) 


class Scratchpad2 0 没有 指定 publie 
( 访问 限定 符 


def private Xx; 
Integer doSstuff'() 


( 


只 自 在 读书 @ 


WWwwW .Zizidiary .com 


198 第 8 章 Groovy: Java 的 动态 伴侣 


沁 三 上 
def 了 def String 2z = "Hello":; 
其 三 3 


} 
} 
继续 语法 精简 的 主题 ， 作 为 一 名 Java 开 发 人 员 ， 你 对 用 在 方法 签名 中 抛 出 已 检查 异常 的 
throws 语 人 句 熟 不 熟悉 ? 


8.3.4 ”异常 处 理 


跟 Java 不 同 ，Groovy 不 区 分 已 检查 异常 和 未 检查 异常 。Groovy 编 译 器 会 忽略 方法 签名 中 的 所 
有 thzrows 博 名 。 

在 保证 源码 可 读 的 前 提 下 ，Groovy 采 用 了 一 些 快捷 语法 来 简化 代码 。 接 下 来 看 一 个 有 严重 语 
义 影 响 的 语法 变化 : 相等 操作 符 。 


8.3.5 ”Groovy 中 的 相等 


遵循 最 小 意外 原则 ，Groovy 把 == 当 做 Java 中 的 equals () 方 法 。 这 是 直觉 式 开 发 人 员 的 又 一 
项 福利 ， 他 们 不 必 再 像 用 Java 时 为 原始 类 型 和 对 象 倒 膳 == 和 equals1)。 

检查 真实 的 对 象 是 否 相 等 ， 需 要 使 用 Groovy 内 置 的 1s() 函数 。 这 一 规则 有 个 例外 ， 就 是 你 
仍然 可 以 用 == 来 检查 一 个 对 象 是 否 为 nul11。 代 码 清单 8-3 说 明了 这 些 特性 。 


代码 清单 8-3 Groovy 中 的 相等 
Integer 其 = new Integer (2) 
Integer YY = new Integer (2) 
Integer z = null 

if 【天 sa 下 | 


| 
println 是 守 == TH 
| 
if (lx.ias(y)) 
| 
println "x is not YI 
} 
if (z.is(null)) 


| 


println "zz is null" 


隐 含 的 equals1() 
调用 


检查 对 象 
是 否 相 等 


用 ie1) 检 查 
是 香 为 mul1l 


if (gs = null} 


| 


println "sz ig null" 


为 null 


当然 ， 如 果 你 育 欢 ， 仍 然 可 以 用 equals |) 方法 检查 相等 关系 。 
最 后 ,还 有 一 个 应 该 简单 提 一 下 的 Java 结 构 一 一 内 部 类 ，Groovy 中 它 基 本 被 一 个 新 的 语言 结 
构 取 代 了 。 
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8.3.6 内 部 类 


Groovy 支 持 内 部 类 ,但 大 多 数 情 况 下 我 们 应 该 用 函数 字面 值 蔡 代 它 。 下 一 节 讨论 函数 字面 值 ， 
它 是 一 个 很 强 的 现代 编程 结构 ， 应 该 占用 更 多 篇 幅 介 绍 。 

用 Groovy 可 以 写 出 更 简洁 的 代码 , 而且 并 不 影响 代码 的 可 读 性 ， 如 果 你 喜欢 , 仍然 可 以 继续 
使 用 Java 的 (大 多 数 ) 语法 结构 。 接 下 来 ， 你 会 看 到 一 些 Java 还 不 具备 的 Groovy 语 言 特性 。 其 中 
的 某 些 特 性 很 可 能 就 是 你 选用 Groovy 的 关键 ， 比 如 XML 处 理 。 


8.4 Java 不 具备 的 Groovy 特性 


Groovy 具 备 一 些 Java 没 有 的 语言 特性 , 起 码 Java 7 还 没有 。 优秀 的 Java 开 发 人 员 就 是 在 这 些 问 
题 上 需要 向 新 语言 求助 ， 和 希望 能 以 更 优雅 的 方式 解决 它们 。 本 节 就 探索 几 个 这 样 的 特性 ， 包 括 : 

口 GroovyBean ， 更 简单 的 bean; 

口 用 操作 符 ? .实现 nul1 对 象 的 安全 访问 ; 

口 猫 王 "操作 待 ( Elvis operator )， 更 短 的 if/else 结 构 ; 

口 Groovy 字 符 串 ， 更 强 的 字符 串 抽象 ; 

口 孙 数 字面 值 ( 即 闭 包 )， 把 晒 数 当做 值 传 递 

口 对 正则 表达 式 的 本 地 支持 ; 

口 更 简单 的 XML 处 理 。 

我 们 会 从 GroovyBean 开 始 , 因为 Groovy 代 码 中 经 常见 到 它们 。 作 为 一 名 Java 开 发 人 员 ， 你 可 
能 有 点 儿 疑 心 ， 因 为 按 JavaBean 的 标准 来 衡量 的 话 ， 它 们 不 太 完 整 。 但 请 你 放心 ，GroovyBean 
很 完整 ， 分 毫 不 差 ， 并 且 用 起 来 更 方便 。 


8.4.1 GroovyBean 


GroovyBean 很 像 JavaBean， 不 过 省 略 了 显 式 声明 的 获取 和 设置 方法 ， 提 供 了 上 自动 构造 方法 ， 
并 人 允许 你 用 点 号 ( . ) 引用 成 员 变 量 。 如 果 需 要 把 某 个 获取 方法 或 设置 方法 设 为 private， 或 者 
希望 改变 默认 的 行为 , 可 以 显 式 声明 那个 方法 ,并 按 你 的 想法 修改 它 。 自 动 构造 方法 只 是 一 个 用 
来 构造 GroovyBean、 传 人 与 GroovyBean 的 成 员 变 量 对 应 的 参数 的 映射 。 

不 论 是 不 群 劳 百 目 己 轩 入 获取 方法 和 设置 方法 ， 还 是 用 IDE 生 成 ， 所 有 这 些 都 省 去 了 我 们 处 
理 JavaBean 时 所 要 编写 的 大 量 合 路 化 代码 。 

我 们 以 一 个 角色 扮演 游戏 ( RPG ) “里 的 character 类 为 例 来 看 一 下 GroovyBean 是 如 何 工作 
的 。 代 码 清 单 8-4 会 输出 srR[18] ，wIS[15] ， 这 是 代表 GroovyBean 力 量 和 智慧 的 成 员 变 量 。 


山 Elvis Aron Presley ( 1935 一 1977 )， 美 国 播 访 乐 史上 影响 力 最 大 的 歌手 ， 有 播 访 乐 之 王 的 准 称 。 一 一 译 者 注 
加 这 里 大 力 推荐 一 下 PCGen ( http://pcgen.sf.net )， 对 于 RPG 粉 来 说 真是 个 非常 好 的 开源 项 目 。 
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lasgs Character 


private int strength 
Private int wisdom 


dat pc = new Character lstrength: 10, wisdom: 15) 
Pe.strength = 18 
println "STR [”+ poe.strength + "] WIS [" + BC.wisdom + "]" 


它 的 行为 跟 Java 里 的 JavaBean 非 党 相似 (封装 性 得 以 保留 )， 而 语法 更 精简 。 


提示 ”可 以 用 BImmutable 注 解 使 GroovyBean 不 可 变 ( 意思 是 它 的 状态 不 可 修改 )。 这 对 于 传递 
线程 安全 的 数据 结构 很 有 有用， 在 并 发 代码 中 用 起 来 更 安全 。 第 10 章 讨论 闭 包 时 我 们 还 会 
进一步 讨论 不 可 变数 据 结 构 的 概念 。 


拉 下 来 我 们 会 转向 Groovy 检 查 nul1 引 用 的 能 力 。 这 会 进一步 减少 套路 化 代码 ， 以 便 你 可 以 
更 快 地 把 想法 变 成 原型 。 


8.4.2 安全 解 引 用 操作 符 


NullpointerException (NPE ) 是 所 有 Java 开 发 人 员 都 挥 之 不 去 的 梦 厅 (很 不 幸 )。 为 了 
避 开 NPE， Java 程 序 员 通 常 都 会 在 引用 对 象 之 前 检查 一 下 它 是 否 为 nul1, 特别 是 在 他 们 不 能 保证 
所 处 理 的 对 象 不 是 nul1 的 情况 下 。 如 果 你 准备 在 Groovy 中 延续 那 种 开发 风格 ， 为 了 遍历 一 个 
Person 对 象 列 表 ， 最 终 编写 的 代码 可 能 像 下 面 这 样 ( 只 是 输出 “Gweneth” )。 

ListcPerson> people = [null, new Person(lname: "Gweneth")] 

for (Person person: people) 1 

if (pareon l= null) | 
println person.getNamel{) 


} 

Groovy 引 人 了 安全 解 引 用 运算 符 ， 用 ? .符号 帮 你 去 掉 一 些 套路 化 的 “如 果 对 象 为 nu11” 检 
查 代码 。 在 使 用 这 个 符号 时 ，Groovy 引 入 了 一 个 特殊 的 null 结 构 ， 表 示 “ 什 么 也 不 做 "， 而 不 是 
真 的 引用 nul1。 

在 Groovy 中 ， 可 以 用 安全 解 引 用 语法 重 写 上 面 的 代码 : 

People = [null, new Perscon (name: "Gweneth"))] 


for (Person person: people) { 
println person? .name 


Groovy 销 数 也 支持 这 种 安全 解 引用 ， 所 以 Groovy 的 默认 集合 方法 ( 比如 max() 方 法 )， 能 自 
动 处 理 好 nul11 引 用 。 


QD Java 最 大 的 憾事 就 是 设 据 实 把 这 个 叫做 wulIReferenceException， 本 书 的 一 位 作者 对 此 一 直 颇 多 她 言 ! 
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接 下 来 是 猫 王 操作 符 ， 看 起 来 和 安全 解 引 用 差不多 ， 但 它 是 用 来 减少 某 些 if/else 结 构 中 的 


8.4.3 ” 猫 王 操作 符 


用 猫 王 操作 符 (?: ) 可 以 把 带 有 默认 值 的 1f/else 结 构 写 得 极其 短小 。 为 什么 叫 猫 王 ”因为 
这 个 符号 看 起 来 明显 很 像 猫 王 易 盛 时 期 梳 的 大 背 头 2。 用 猜 王 操作 符 不 用 检查 nu11， 也 不 用 重复 
变量 。 

假设 你 要 检查 王牌 大 贱 认 是 不 是 活跃 的 侦探 。 在 Java 中 可 能 要 用 三 元 操作 符 ; 


String agentStatus = "Active"; 
String statuis = agentStatus l= null ? agentStatus : "Inactive"; 


Groovy 能 缩短 这 个 语句 ， 是 因为 它 能 在 需要 时 将 类 型 强制 转换 为 boolean， 比 如 if 语句 的 
条 件 判 断 ,。 在 前 面 的 代码 中 ，Groovy 把 string 转 换 为 boolean, 假如 string 是 null, 它 会 被 转 
换 成 Boolean 值 false， 所 以 可 以 省 略 null 检 查 。 因 而 前 面 的 代码 可 以 写成 这 样 : 


String agentSstatus = "Active" 
String status = agqentSstatus ? agqentSstatus : "Tnactive" 


但 这 样 还 是 要 重复 agentStatus 变 量 ，Groovy 可 以 让 我 们 不 再 重复 输入 。 用 和 猜 王 操 作 符 可 
以 去 掉 重 复 的 变量 名 : 


string agentSstatus = "Active" 
String status = AagentSstatus ?: "Inactive" 


第 二 个 agentstatus 没 了 了， 代码 更 简洁 了 。 
好 了 ， 现 在 该 去 看 看 Groovy 字 符 串 了， 看 看 它们 跟 Java 和 常规 string 有 什么 不 同 。 


8.4.4 ”增强 型 字符 串 


Groovy 有 一 个 String 类 的 扩展 类 Gstring， 它 比 Java 中 标准 的 String 强 ， 也 更 灵活 。 
尽管 双 引 号 也 有 有效， 但 按照 惯例 ， 普 通 字 符 串 是 用 开 闭 两 个 单 引 号 定义 的 。 比 如 ; 


string ordinarystring = 'ordinary string' 
string ordinarySstring2 = "ordinary string 2" 


而 GString 必 须 用 双 引 号 定义 。 对 于 开发 人 员 来 说 , 使 用 它 最 大 的 好 处 是 可 以 包含 可 在 运行 
时 计算 的 表达 式 ( 用 ${) ) 如 末 Gstring 随 后 被 转 为 普通 宇和 侍 串 ( 比如 传 给 了 println )， 
GString 中 的 表达 式 虱 会 饼 霄 换 为 其 计算 箔 米 。 比 如 : 

String name = 'Gweneth' 


def dist = 3*2 : 
string crawling = "S${name} is crawling ${dist} feet!" 


其 中 的 表达 式 计 算 后 被 转 到 可 以 调用 tostring() 的 Object 上 ， 或 是 函数 字面 值 上 。( 请 参 
风 http://groovy.codehaus.org/Stringstand+GString 了 和 解 关 于 函数 字面 值 和 Gstring 规 则 的 细节 ,) 


册 本 书 的 作者 都 郑重 声 明 ， 我 们 根本 不 知道 猫 王 在 鼎盛 时 期 长 什么 样 。 我 们 真 没 那么 老 ， 不 开玩笑 ! 
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警告 GString 的 底层 并 不 是 Java 中 的 Stringl 尤其 不 应 该 把 GSEting 作 为 映射 中 的 键 ， 或 者 
比较 它们 是 否 相 等 。 结 果 是 不 可 预料 的 ! 


Groovy 中 男 一 个 有 点 儿 用 的 结构 是 三 引号 string 或 三 引号 Gstring， 它 们 可 以 在 源码 中 定 
义 跨行 字符 帅 。 

"This GString 

wraps OVver two lines!"™"" 


接 下 来 我 们 要 疝 函数 字面 值 进军 了 。 由 于 最 近 几 年 业内 兴起 了 对 函数 式 语言 的 兴趣 ,这 个 编 
程 技 蕊 也 成 了 一 个 热门 话题 。 要 弄 懂 陋 数字 面值 ， 可 能 需要 动 动脑 筋 。 如 果 你 没 用 过 ， 也 就 是 说 
如 果 这 是 你 第 一 次 用 ， 也许 你 现在 就 该 先 起 身 将 公国 杯 加 满 自 己 育 欢 的 饮品 。 


8.4.5 ”函数 字面 值 


函数 字面 值 表 示 一 个 可 以 当做 值 传 递 的 代码 块 , 也 可 以 像 操 作 任何 值 一 样 操作 。 可 以 当 参 数 
传 给 方法 ， 可 以 给 变量 赋值 ， 等 等 。 这 个 语言 特性 已 经 成 为 Java 社 区 的 讨论 热点 ， 但 对 于 Groovy 
程序 员 来 说 ， 它 们 是 标 配 的 工具 。 

举例 说 明 向 来 都 是 学 习 新 概念 的 最 好 方法 ， 我 们 先 来 看 几 个 例子 吧 ! 

假设 我 们 有 一 个 普通 的 静态 方法 ， 要 构建 一 个 stzing 来 向 作者 或 读者 问好 。 我 们 用 常规 方 
式 从 这 个 类 的 外 部 调用 该 方法 ， 如 代码 清单 8-5 所 示 ， 
代码 清单 8-5 “一 个 简单 的 静态 函数 

class StringUtils | 

| 静态 方法 声明 


static String sayHellol!string name) < 


if (name == "Martijn" || name == "Ben") 
"Hello author ™ + name + "I" 

el se 
"Hello reader " + name + "ll" 


| 
SEE | | 调用 者 
println StringUtils.savHellol"Bob"):; 
有 了 函数 字面 值 , 你 不 用 方法 或 类 结构 也 可 以 实现 同样 的 功能 ， 只 要 把 代码 放 在 函数 字面 值 
里 。 而 函数 字面 值 又 可 以 赋值 给 一 个 变量 ， 从 而 可 以 被 传递 和 执行 。 
代码 清单 8-6 把 函数 字面 值 赋 值 给 sayHello, 传 人 参数 "Marcijn" ,并 最 终 输出 “Hello author 
Martijn! 。 
代码 清单 8-6 ”使 用 简单 的 函数 字面 值 


def sayHello = 可 一 一 一 一 一 一 一 一 涌 数 字面 值 赋值 


name -> 
if {name == "Martijn" || name == "Ben") 
"Hellso autheor " #4 name + 1" 
else 变量 与 处 理 逻 辑 分 开 
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"Hello reader " + name 十 "I" 


[ 
println(sayHello("Martijn")) 


注意 函数 字面 值 开始 处 的 {。 把 传人 函数 字面 值 的 参数 跟 处 理 逻辑 分 开 的 箭头 ( -> ) 全。 最 
后 是 郴 数 字面 值 纺 束 处 的 ] 。 

在 代码 清单 8-6 中 ， 函 数字 面值 的 定义 方式 非常 像 方法 的 定义 方式 。 因 此 你 可 能 在 想 :“ 它 们 
看 起 来 也 不 是 特别 有 用 !” 只 有 开始 用 它们 创作 ( 用 函数 方式 思考 ) 时 ， 你 才能 真正 发 现 它们 的 
好 ， 比 如 说 跟 Groovy 对 集合 的 内 置 支 持 结合 起 来 之 后 ， 函 数字 面值 会 特别 强大 。 


8.4.6 内置 的 集合 操作 


Groovy 有 几 个 可 以 用 于 集合 ( 列表 和 映射 ) 的 内 置 方法 。 这 种 在 语言 层面 对 集合 的 支持 ， 跟 
函数 结合 在 一 起 ， 可 以 极 大 减少 程序 员 在 Java 中 必 写 的 那些 套路 化 代码 ; 并 且 代 码 仍然 很 容易 看 
懂 ， 不 影响 维护 。 

表 8-1 是 一 些 使 用 了 上 数字 面值 的 内 置 国 数 。 


表 8-1 ”Groovy 中 的 部 分 集合 函数 

方 法 描 述 

each ”遍历 集合 ， 对 其 中 的 每 一 项 应 用 函数 字面 值 

collect 收集 在 集合 中 每 一 项 上 应 用 函数 字面 值 的 返回 结果 (相当 于 其 他 语言 map/reduce 中 的 map 函 数 ) 
inject 用 函数 字面 值 处 理 集合 并 构建 返回 值 (相当 于 其 他 语言 里 mapireduce 中 的 reduce 函 数 ) 
findAll 找到 集 各 中 所 有 与 函数 字面 值 号 配 的 元 素 

max 返回 集合 中 的 最 大 值 

min 返回 集合 中 的 最 小 值 


Java 编 程 过 程 中 遍历 集合 ， 并 对 其 中 每 个 对 象 执行 某 种 操作 是 很 常见 的 任务 。 比 如 说 ， 如 果 
你 想 在 Java 7 中 输出 电影 名 称 ， 很 可 能 会 写 出 如 代码 清单 8-7 所 示 的 代码 : 


代码 清单 8-7 ”在 Java 7 中 输出 一 个 集合 
List<String> movieTitles = new ArravyList<>(); 
movieTitles.add(l"Seven"): 
movieTitles.add("Ssnow White").; 
movieTitles.add{"Die Hard'"):; 


for lString movieTitle : movieTitles) 
{ 
Syastem.out .println(movieTitlel.,; 


| 
Java 中 有 帮 你 少 写 代码 的 技巧 ， 但 不 管 怎 样 都 要 用 某 种 循环 结构 手工 遍历 电影 名 称 的 List。 
在 Groovy 里 可 以 用 内 置 的 集合 遍历 功能 (each 函数 ), 并 且 函 数字 面值 可 以 减少 大 量 你 需要 


不 ,我们 可 不 会 告诉 你 谁 育 欢 《白雪 公主 》( 反正 不 是 我 俩 
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自己 编写 的 代码 。 此 外 , 这 样 还 能 反 转 列表 和 所 要 执行 的 算法 之 间 的 关系 。 不 再 是 把 集合 传递 到 
方法 中 ， 而 是 把 方法 传人 到 集合 中 ! 

下 而 的 代码 和 代码 清单 8-7 所 做 的 工作 完全 一 样 ， 但 只 有 短 短 的 两 行 ， 很 容 多 读 异 : 

movieTitles = ["Seven®", "SnowWhite", "Die Hard"] 

movieTitles.each({x -> println Xl) 


实际 上 ， 如 果 使 用 隐 含 的 it 变量 ， 这 段 代码 还 可 以 变 得 更 精简 ，it 变 量 可 以 用 在 单 参 的 函 
数字 面值 中 ， 代 码 如 下 所 示 ”: 


movieTitles = ["Seven®", "SnowWNhite", "Die Hard")] 
movieTitles.each({println it}) 


看 ， 这 段 代码 简洁 易 读 ， 并 且 效 果 和 Java 7 那个 版 本 一 样 。 


提示 。 只 能 介绍 这 么 多 了 ， 如 果 你 想 研 究 更 多 例子 ,推荐 你 到 Groovy 的 网 站 上 去 看 看 与 集合 相 
关 的 内 容 ( http://groovy.codehaus.org/JN1015-Collections ), 或 者 读 读 Dierk Kinig, Guillaume 
Laforge、Paul King、Jon Skeet 和 Hamlet D'Arcy 会 著 的 Groovy in Action, second edition 
( Manning, 2012 )， 这 是 一 本 相当 不 错 的 书 。 


下 一 个 语言 特性 是 Groovy 内 置 的 正则 表达 式 支 持 , 你 可 能 要 花 点 儿 时 间 才 能 见 悉 ,所 以 信者 
咖啡 动 儿 ， 我 们 赶 案 来 看 看 吧 ! 


8.4.7 ”对 正则 表达 式 的 内 置 支 持 


Groovy 把 正则 表达 式 当 做 语言 的 一 部 分 ， 所 以 用 Groovy 处 理 文本 要 比 Java 简 单 得 多 。 表 8-2 
中 是 Groovy 可 用 的 正则 表达 式 语 法 ， 以 及 Java 要 之 对 应 的 东西 。 
表 8-2 ” Groovy 正则 表达 式 语法 
方 法 描述 及 Java 中 的 对 等 物 
~ 创建 一 个 模式 (创建 一 个 编译 的 Java Pattern 对 象 ) 
a 创建 一 个 匹配 露 (创建 一 个 Java Matcher 对 象 ) 
机 ， 计 算 字符 串 (相当 于 在 Pattern 上 调用 Java match1) 方 法 ) 


假设 你 从 一 个 人 硬件 上 收 到 了 一 些 日 志 数 据 ， 要 部 分 匹配 其 中 一 些 错 误 日 志 。 比 如 查找 模式 
1010 的 实例 ， 然 后 再 找 0101。 在 Java 7 中 ， 实 现代 码 可 能 如 下 所 示 。 

Pattern pattern = Pattern.compile("l1010"),; 

String input = "1010"; 

Matcher nmatcher = pattern.matcher (1nput).:; 

if (input .matches("1010")) 


input = matcher.replaceFirst ("0101"); 


DD Groovy 高 手 会 说 实际 上 还 可 以 简化 ,一 行 足 类 
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System.out .printlnlinput}); 


在 Groovy 中 ， 每 行 代码 都 变 短 了 ， 因 为 Pattern 和 Matcher 对 象 是 内 置 在 语言 中 的 。 当 然 ， 
输出 ( 0101 ) 还 和 原来 一 样 ， 请 看 代码 。 


def pattern = /1010/ 

def input = "i1010" 

def matcher = input == pattern 
if (input ==» pattern) 


| 
input = matcher.replaceFirst ("0101") 
println input 


Groovy 六 持 完 整 的 正则 表达 式 语 义 ， 所 采用 的 方式 和 Java 一 样 ， 所 以 你 熟悉 的 那 种 灵活 性 还 
人 在。 

正则 表达 式 跟 明 数 字面 值 结合 得 也 很 好 。 比 如 分 析 Stzing 得 到 一 个 人 的 名 字 和 年 龄 ， 并 输 
出 详细 信息 。 


("Hazel 1" == /(\w+) (\d+)/) .each {full, name, age 
-> println "$name is $age Years old.") 


或 许 你 应 该 借 这 个 机 会 稍微 放松 一 下 , 接 下 来 我 们 马上 就 要 探索 一 项 完全 不 同 的 技术 : XML 
处 理 。 
8.4.8 简单 的 XML 处 理 


Groovy 有 构建 器 的 概念 ,用 Groovy 原 生 语 法 可 以 处 理 任何 树 型 结构 的 数据 ,包括 HTML XML 
和 JSON。Groovy 理 解 开 发 人 员 想 轻松 处 理 这 种 数据 的 需求 ， 所 以 提供 了 开 箱 即 用 的 构建 器 。 


a mi 站 ra | EE LW = a ne L 
省 | 抽 ei a | WE 
sah 可 一 -一 rm ls | wr 


Wy = Ce 六 和 ML 


A yop pT ry : 
因为 软件 开发 人 员 已 经 把 XML 当成 编程 语言 来 用 了 ， 可 它 不 是 图 灵 完备 的 语言 ， 所 以 它 不 过 
合 干 这 些 事 。 希 望 XML 能 在 你 的 项 目 中 得 其 所 哉 ， 只 是 用 来 交换 数据 


本 节 重 点 是 XML, 一 种 常用 的 交换 数据 格式 。 尽 管 Java 语 言 的 核心 ( 通过 JAXB 和 JAXP ) 以 
及 浩 浩荡 菏 的 第 三 方 类 库 ( XStream 、Xerces、Xalan 等 ) 组 成 了 庞大 的 XML 处 理 大 军 ， 但 选 哪 个 
方案 经 常 让 人 难以 抉择 ， 并 且 采 用 相应 方案 的 Java 代 码 会 变 得 非常 元 长 。 

本 节 会 带 你 用 Groovy 创 建 XML ， 并 告诉 你 如 何 把 XML 解析 为 GroovyBean。 

1. 创建 XML 

用 Groovy 构 建 XML 文档 非常 简单 ， 比 如 person， 


山 对 于 一 种 语言 来 说 ， 如 果 是 图 灵 完 备 的 ， 那 它 至 少 必 须 能 做 条 件 分 支 判 断 ， 并 能 修改 内 存 数据 。 
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<person id='2'> 
<name>Gwenethea /names 
<age>l</ ge> 
</person> 


Groovy 能 用 内 置 的 MarkupBuilaer 产 生 这 个 XML。 产 生 personXML 记 录 的 代码 如 代码 清 
单 8-8 所 示 : 


代码 清单 8-8 产生 简单 的 XML 
def writer ss new StringWriter') 
def xm] = new groovy .xml .MarkupBuilder (writer) 
Xml person(id:2) 1{ 
name 'Gweneth' 
age 1 
| 
println writer.toStringl() 
注意 看 person 的 起 始 元 素 ( 属性 ia 设 置 为 2 ) 创建 起 来 多 么 简单 ， 根 本 不 用 定义 Person 对 
象 。Groovy 不 会 强迫 你 显 式 地 弄 一 个 GroovyBean 来 文 撑 XML 的 创建 , 青 一 次 广 省 了 时 间 和 精力 。 
代码 清单 8-8 中 的 例子 相当 简单 。 你 可 以 多 做 些 试验 ， 把 输出 类 型 stringwriter 改 掉 ， 并 
上 且 可 以 尝试 用 不 同 的 构建 器 ， 比如 groovy .json.JsonBuilder () ,即刻 创建 JSON”, 在 处 理 更 
复杂 的 XML 结构 时 ， 命 名 空间 和 其 他 特定 构造 的 处 理 上 也 有 额外 的 辅助 方法 。 
你 可 能 还 希望 执行 反 癌 操作 ， 读 取 XML 并 把 它 解 析 成 GroovyBean。 
2. 解析 XML 
Groovy 有 几 种 解析 XML 输 入 的 办 法 。 表 8-3 到 出 了 其 中 三 个 方法 ， 这 是 从 Groovy 的 官方 文档 
( http://docs.codehaus.org/display/GROOVY/Processing+XML ) 中 拿 过 来 的 。 
表 8-3 Groovy XML 解析 技术 
方 ”法 描 述 
XMLParser 支持 XML 文 档 的 GPath 表 杰 式 
XMLSlurper 跟 XMLFarsez 类 似 ， 但 以 情 加 载 的 方式 工作 
DOMCategory 用 一 些 语 革 支持 DOM 的 底层 解析 


这 三 个 用 起 来 郡 很 创 单 ， 但 这 一 市 我 们 主要 关心 XMLParser 的 用 法 。 


注意 GPath 是 一 种 表达 式 语 言 。Groovy 文 档 (http://groovy.codehaus.org/GPath ) 中 有 它 的 全 部 
内 容 。 


我 们 把 代码 清单 8-8 中 产生 的 那个 表示 “Gweneth”( 人 名 ) 的 XML 拿 过 来 ， 并 把 它 解析 到 一 
个 GroovyBean Person 中 ， 如 代 公 清单 8-9 所 示 。 


山 关于 这 一 问题 ，Dustin 在 他 的 博客 Inspired by Actual Events ( http://marxsoftware.blogspol.com/ ) 上 有 一 篇 视 梭 的 文 
章 ， 标 题 是 “Groovy 1.8 Introduces Groovy to JSON "。 
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代码 清单 8-9 ”用 xMLParser 解 析 : 


class XmlExample | 
Static def PERSON = 


<person id='2'> 六 ML 作 为 Groovy 源 码 
<name>sGwenetha</names 
<aAge>l</age> 

< /person> 

但 到 师 


} 
class Person {def id; def name; def agel 村 

ge| | Groovy 中 的 
def xmlPerson = new XmlParser!{). Parson 定 义 


parseText (XmlExample . PERSON) Ss | 
@ 读 取 XML 


Berson Bp = new Pergonlid: xmlPerson .®id, 


name: xmlPerson.name.text(), Ro - 
age;:; XMmlPerBon.age.text|())} | [3) 填 入 GroovyBean 
Person 中 


printiln "${p.id}, ${p.name}, ${p.age}" 

我 们 一 开始 抄 了 点 儿 近 路 ， 把 XML 文 档 直 接 放 在 代码 中 了 ， 这 样 它 就 会 出 现在 cLASSsPATH 
中 人力 。 直 正 的 第 一 步 是 用 xMLParser 中 的 parseText |) 方 法 读 取 XML 数 据 全 。 然 后 创建 新 的 
Person 对 象 ， 给 它 赋 值 个 ， 最 后 输出 Person， 以 便 你 能 用 肉眼 检查 一 下 。 

我 们 对 Groovy 的 介绍 到 此 就 完成 了 。 现 在 ， 你 可 能 觉得 心里 痒痒 的 ， 想 在 自己 的 Java 项 目 里 
使 用 一 些 Groovy 特 性 | 下 一 节 ， 我 们 会 市 你 看 看 Java 如 何 跟 Groovy 互 操作 ，。 由 此 你 将 迈 出 作为 优 
秀 Java 开 发 者 的 重要 一 步 : 成 为 一 名 JVM 多 语言 程序 员 。 
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这 一 节 很 短 , 但 不 要 低估 它 的 重要 性 ! 如 果 你 是 按 顺序 看 到 这 里 的 , 这 里 就 是 你 要 跳 的 龙门 ， 
跳 过 去 你 就 不 是 只 在 JVM 上 做 Java 开 发 的 人 了 。JVM 上 有 多 种 语言 作为 Java 的 补充 ， 优 秀 的 Java 
开发 者 要 具备 使 用 它们 的 能 力 ，Groovy 是 个 很 好 的 起 点 1 

首先 ， 你 会 重 温 一 下 从 Groovy 中 调用 Java 是 多 么 简单 。 之 后 你 会 看 到 Java 与 Groovy 交 互 的 三 
种 常用 途径 ， 使 用 Groovyshe11、Gro ovyClassLoader 利 GroovyScriptEngine。 

我 们 先 来 重 温 一 下 Groovy 里 怎么 调用 Java。 


8.5.1 从 Groovy 调用 Java 


还 记得 吗 ? 我 们 错过 从 Groovy 调 用 Java 很 简单 ， 你 只 要 把 JAR 放 到 cLassPaTH 中 ， 人 然后 用 标 
准 的 ijmport 语 句 就行 了 。 这 儿 有 个 例子 ， 引 入 流行 的 Joda 日 期 时 间 类 库 中 org .joda .time 包 里 
的 类 ， 


import org.joda.time.*; 


Dj 在 Java gs 发 布 之 前 ， 实 际 上 Joda 一 直 都 不 是 Java 的 标准 日 期 时 间 类 库 。 
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可 以 跟 在 Java 中 一 样 使 用 这 些 类 。 下 而 的 代码 会 输出 当前 月 份 的 数值 表示 | 


DateTime dt = new DateTime!) 
int month = dt .getMonthOfYear (| 
println month 


哦 ， 肯 定 要 比 这 个 更 复杂 点 儿 吧 ? 

阿 克 巴 上 将 :“ 陷 阱 !”” 

开 个 玩笑 , 这 里 没什么 陷阱 ! 真 的 就 这 么 简单 , 那 我 们 是 不 是 应 该 看 看 更 困难 的 情况 ? 从 Java 
调用 Groovy 并 得 到 有 意义 的 结果 还 是 有 点 儿 技 术 含 量 的 。 


8.5.2 从 Java 调用 Groovy 


从 Java 程 序 调用 Groovy 需 要 把 Groovy 及 其 相关 的 JAR 放 到 这 个 程序 的 cLASSPATH 下 ， 因 为 它 
们 都 是 运行 时 依赖 项 。 


提示 “只 需要 把 GROOVY HOME/embeddable/groovy-all-1.8.6.jar 文 件 放 到 CLASSPATH 中 。 


下 面 是 几 种 从 Java 调 用 Groovy 代 码 的 办 法 ; 

口 使 用 Bean Scripting Framework ( BSF )， 即 JSR 223: 

口 使 用 GroovyShell; 

口 使 用 GroovyClassLoader; 

使 用 GroovyscriptEngine:; 

口 使 用 概 人 人 式 Groovy 控 制 台 。 

我 们 在 这 一 节 重 点 讨论 最 和 营 用 的 办 法 (GroovyShell、GroovyClassLoader 和 Groovy- 
ScriptEngine )。 先 从 最 简单 的 Groovyshell 开 始 。 

1. GroovyShell 

在 临时 性 快速 调用 Groovy 并 计算 表达 式 或 类 似 于 脚本 的 代码 时 ， 可 以 用 Groovyshell。 上 比 
如 说 ， 有 些 开 发 人 员 可 能 更 喜欢 用 Groovy 做 数值 处 理 ， 就 可 以 调用 Groovyshel1 执 行 一 些 数 学 
计算 。 代 码 清 单 8-10 会 返回 用 Groovy 的 数值 相 加 得 到 的 结果 10.4。 


代码 清单 8-10 ”在 Java 中 用 Groovysheli 执 行 Groowy 代 码 
import groovy. lang.GroovySsShell]l; 
import groovy .lang.Binding:; 
import java.math.BigDecimal; 


public class UseGroovyShell 1 


public static void mainlstring[] args}) | 
加 星 战 迷 对 这 句 在 网 上 广 为 流 传 的 话 应 该 不 会 感到 陌生 。 
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Binding binding = new Binaino() ; 


binding.setVvariable ("x", 2.4); | 设置 ahell1 上 
binding.setVariable(l"y", 8); 的 bindain 
GroovySshell shell = new GroovyShell (binding):; ; 
Object value = shell.evaluate{("x + y"); 天 
asBert value.equals (new BigDecimal (10.4)); 计算 井 返 回 

| 表达 式 


| 

用 Groovyshe11 只 能 应 付 快速 执行 小 段 Groovy 代 码 的 情况 ， 如 果 要 与 一 个 完整 的 Groovy 类 
殉 互 ， 该 怎么 办 呢 ? 这 时 可 以 用 GroovyClassLoader。 

2. GroovyClassLoader 

从 开发 人 员 的 角度 看 ，GroovyclassLoader 的 表现 很 
要 调用 的 方法 ， 然 后 调用 就 行 卫 。 

下 面 的 代码 中 有 一 个 简单 的 calculateMax 类 ， 其 中 有 个 getmax 方 法 ， 会 使 用 Groovy 内 置 
的 max 阴 数 。 要 在 Java 里 通过 GroovyClassLoader 运 行 这 个 方法 ， 需 要 用 下 面 的 代码 创建 一 个 
Groovy 文 件 ( CalculateMax.groovy ): 

class CalculateMax 1 

def Integer getMax(List values) | 
values.max(); 

, } 

现在 我 们 有 了 要 执行 的 Groovy 脚 本 ， 可 以 从 Java 调 用 它 了 了 。 在 代码 请 单 8-11 中 ， 从 Java 调 用 
CalculateMax getMax 上 图 数 ， 返 回 了 传人 参数 中 的 最 大 值 10。 


代码 清单 8-11 在 Java 中 用 GroovyclassLoader 执 
import java.io.File; 
import java.io.IOException; 准备 
import java.util .ArrayLiast; 
import groovy .lang.GroovyClassLoader.; 
import groovy. lang.GroovyObject:; 
import org.codehaus.groovy .control .CompilationFailedException,; 


象 JavahJCcClassLoader。 找 到 类 和 想 


public class UseGroovyClassLoader | 
public static void main(String[] argsl 1 得 到 类 
GroovyClassLoader loader = new GroovyClassLoader |(); 


try { 
Class<?> groovyClass = loader parseClass | 
new File'"CalculateMax .groovy")).; 


GroovyQbject groovyObject = (IGroovyObject) ee 


groovyClass.newInstance't{).; 


ArrayList<Integer> nmbers = new ArrayList<s(); 
numbera.addlnew Integer(1})); se 
numbera.addlnew Integer(10)); 
Object [] arguments = Inumbers}; - 
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Obiject VBLUe = 调用 Groovy 方 法 
groovyObject .invokeMethodl"getMax", arguments):; ,| 


asgSert value.eqguales (new Integer (10)); 


cateh (CompilationFailedException | IOExcaption | InstantiationException 
| IllegqalAccessException e) | 
System.out .printlnle.getMessage(})}):; 
| 
| 


这 种 技术 在 调用 几 个 Groovy 实 用 类 时 可 能 会 有 用 。 但 如 果 要 用 大 量 的 Groovy 代 码 , 我 们 推荐 
使 用 完整 的 G6roovysScriptEngine。 

3. GroovyScriptEngine 

使 用 GroovyscriptEngine 要 指明 Groovy 代 码 的 URL 或 所 在 目录 。Groovy 肢 本 引擎 会 加 载 
那些 脚本 ， 并 在 必要 时 进行 编译 ， 包 括 其 中 的 依 环 脚本 。 上 比如 说 你 修改 了 脚本 B， 而 脚本 A 依 来 
于 B， 则 引擎 会 全 重新 编译 它们 。 

假设 有 一 个 Groovy 肢 本 ( Hello.groovy ) 证 义 了 一 个 简单 的 “Hello” 博 句 ， 后 面 跟着 一 个 名 
字 (要 从 Java 应 用 程序 中 传人 的 参数 )。 

def helloStatement = "Hello $[name}" 

然后 Java 程 序 会 通过 GroovyScriptEngine 使 用 Hello.groovy， 并 输出 一 句 问 候 ， 如 代码 清 
单 8-12 所 示 : 


代码 清单 8-12 ”在 Java 中 用 GroovyscriptEngine 执 行 Gr 


import groovy .lang.Binding:; 

import groovy .util ,GroovyScriptEngine; 

import groovy.util.ResourceBxception,; 

import groovy .util] .ScriptExcept1ion,; 

import Java.,1ios. TIOExcept1ion,; ， 
设置 根 目录 


public class UseGroovyScriptEngine | 
public static void mainlSstring[] args) 
[ 
try 1 : 
3tring[] roots = new String[] {1"/src/main/groovy"|]; 叶 一 梓 始 化 引擎 
GroovyScriptEngine gse = 
new GroovySscriptEngine (FotBsl 


Binding binding = new Binding():; 


binding.setVariable("name", "Gweneth"); 运行 脚本 
Object output = gse.runl"Hello.groovy", binding).:; | 


agBert output .equals(l"Hello Gweneth"):; 

| 

catch (IOEXCeDptic | ResourceBException | ScriptException e) I 
System.Gout .println(le.getMessage(l)),; 


1 地 
OEEEE 


8.6 小 结 211 


GroovyScriptEngine 监 控 之 下 的 任何 Groovy 脚 本 都 可 能 被 程序 员 一 时 兴起 改 掉 o 比如 说 ' 
将 Hello.groovy 改 成 这 样 ; 

def helloStatement = "Hello ${(name}, it's Groovy baby, yeah!™ 

这 段 Java 代 码 下 次 再 运行 时 ， 它 就 会 用 这 个 新 的 ， 更 长 的 消息 。 这 样 Java 应 用 程序 就 具备 了 
以 前 根本 不 可 能 出 现 的 动态 灵活 性 。 这 在 某 些 情况 下 简直 是 无 价 之 宝 ， 比如 调试 生产 环境 下 的 代 
码 ， 在 运行 时 修改 系统 属性 ， 还 有 很 多 …… 

至 此 ， 对 Groovy 的 介绍 真 要 结束 了。 我 们 已 经 走 了 很 还 了 1 


8.6 小结 


Groovy 有 多 种 引 人 注 目的 特性 ,这 使 它 成 为 一 门 可 以 和 Java 共 用 的 出 色 语 言 ,你 可 以 用 和 Java 
非常 相近 的 霹 法 ， 也 可 以 用 更 精简 的 代码 实现 相同 的 逻辑 。 这 种 精简 并 不 以 牺牲 可 读 性 为 代价 ， 
而 且 Java 开 发 人 员 在 采用 跟 集 合 、nu11 引 用 处 理 和 GroovyBean 相 关 的 新 语法 时 不 存在 什么 困难 。 
然而 ，Groovy 给 Java 开 发 人 员 设 了 几 个 陷阱 ,但 你 已 经 搞定 了 大 多 数 情 况 ， 希望 你 能 带领 同事 走 

很 多 Java 开 发 人 员 都 对 Groovy 中 的 几 个 语言 特性 感到 眼 多 ， 希望 有 朝 一 日 Java 语 言 中 也 能 有 
这 些 特性 。 其 中 最 难 掌 握 、 也 最 强大 的 就 是 函数 宇 面值 , 它 是 一 种 能 在 集合 上 轻松 进行 操作 的 强 
大 编程 技术 跟 其 他 技术 一 起 )。 当 然 ， 集 合 享受 的 是 一 等 公民 的 待遇 ， 你 能 用 更 短小 易 用 的 语 
法 来 创建 、 修 改 和 操作 它们 。 

大 多 数 Java 开 发 人 员 都 要 在 Java 程 序 里 生成 或 解析 XML ， 对 此 Groovy 也 能 助 你 一 臂 之 力 , 它 
能 用 内 置 的 XML 支持 帮 你 挑 起 大 部 分 重担 。 

借助 各 种 技术 把 Java 代 码 和 Groovy 代 码 集成 在 一 起 解决 编程 问题 ,你 已 经 向 多 语言 程序 员 葬 
出 了 一 步 。 

我 们 的 Groovy 旅 程 还 没有 结束 。 在 第 13 章 讨论 快速 Web 开 发 时 ,还 有 更 多 的 Groovy 特 性 等 着 
我 们 去 使 用 和 探索 。 

接 下 来 ， 我 们 请 出 Scala， 另 外 一 门 已 在 业内 造成 小 麦 动 的 JVM 语 言 。Scala 既 是 面向 对 象 语 
症 ， 也 是 隙 数 式 语言 ， 要 解决 现代 编程 中 进退 两 难 的 问题 ，Scala 是 值得 一 看 的 语言 。 
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本 章 内 容 

口 Scala 不 是 Java 

口 Scala 语 法 及 更 加 本 数 化 的 风格 
口 match 表 达 式 与 模式 

口 Scala 的 类 型 系统 和 集合 

口 Scala 并 发 之 actor 


Scala 是 出 自学 术 界 和 编程 语言 研究 社区 的 语言 。 由 于 其 强大 的 类 型 系统 和 先进 特性 , 又 有 精 
英 团 队 证 明了 其 价值 ， 它 已 经 赢得 了 一 定数 量 开发 者 的 青睐 。 

现在 Scala 里 有 很 多 有 意思 的 地 方 ， 但 要 评判 它 能 否 完 全 渗入 Java 生 态 系 统 ， 以 及 能 否 挑 战 
Java 语 言 的 霸主 地 位 ， 还 为 时 尚 早 。 

最 合理 的 预测 是 Scala 会 被 更 多 的 团队 接受 , 并 最 终 被 项 目 采 纳 。 在 接 下 来 的 三 四 年 中 , 我 们 
预计 会 有 大 量 开 发 人 员 在 项 目 中 见 到 Scala 的 身影 。 也 就 是 说 作为 优秀 的 Java 开 发 者 ,你 应 该 了 解 
它 ， 能 判断 出 它 是 否 适 用 于 自己 的 项 目 。 


例子 ”金融 风险 模拟 应 用 可 能 需要 采用 Scala 新 颖 的 面向 对 得 方式 、 类 型 推断 、 灵 活 的 语法 、 新 
的 集合 类 ( 包括 自然 的 函数 式 编程 风格 ， 上 比如 映射 /过 滤器 惯用 语 )， 以 及 基于 actor 的 并 
发 烧 型 。 


对 于 Java 开 发 人 员 来 说 ， 刚 开始 接触 Scala 时 最 应 该 牢记 于 心 的 是 : Scala 不 是 Java。 

这 看 起 来 似乎 是 显而易见 的 。 毕 况 ， 每 种 语言 都 不 同 ，Scala 当 然 也 和 Java 不 一 样 。 但 我 们 在 
第 7 章 提 到 过 ， 有 些 语言 非常 相似 。 第 8 但 介绍 Groovy 时 也 在 强调 它 和 Java 相 似 的 地 方 。 希 望 这 些 
内 容 在 你 首次 探索 Groovy 这 门 非 Java JYMi 语 言 时 能 够 对 你 有 所 帮助 。 

本 章 我 们 想 做 点 不 一 样 的 事情 , 先 突出 Scala 中 一 些 非常 独特 的 语言 特性 。 我 们 喜欢 把 这 上 比喻 
成 “Secala 的 宜 居 之 所 ”， 告 诉 你 怎么 写 Scala 代 码 才 不 会 像 是 从 Java 翻 译 过 来 的 。 之 后 我 们 会 阐述 
项 目 问题 ， 搞 明白 Scala 是 不 是 适用 于 你 的 项 目 。 然 后 ,我 们 看 一 些 Scala 在 语法 上 的 创新 ， 这 些 
创新 让 Scala 代 码 变 得 即 简洁 又 漂亮 。 接 下 来 是 Scala 处 理 面 向 对 象 的 方式 ， 然 后 是 一 节 介 绍 集合 
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和 数据 结构 的 内 容 。 最 后 收尾 的 一 节 是 关于 Scala 并 发 和 它 强 大 的 actor 模 型 的 内 容 。 

在 讨论 Scala 的 独特 性 时 , 我 们 会 一 并 解释 它 的 语法 ( 以 及 其 他 必要 的 概念 )。 跟 Java 比 ，Scala 
是 一 门 相当 庞大 的 语言 , 需要 掌握 的 基础 概念 和 语法 点 更 多 。 这 就 是 说 你 要 做 好 心理 准备 , 随 着 
接触 到 的 Scala 代 码 越 来 越 多 ， 你 需要 自己 花 时 间 和 精力 去 更 多 地 探索 这 门 语言 。 

我 们 先 来 大 体 看 一 下 后 面 会 遇 到 的 一 些 主题 。 这 能 帮 你 熟悉 Scala 不 同 的 语法 和 思维 方式 , 并 
各 你 打下 基础 ， 以 便 更 好 地 学 习 新 知识 。 


9.1 走马 观 花 Scala 


下 面 是 我 们 准备 展示 的 主要 内 容 : 

口 Scala 语 言 的 精炼 ， 包 括 类 型 推断 的 能 力 ; 

口 match 表 达 式 ， 以 及 模式 和 case 类 等 相关 概念 ; 

口 Scala 的 并 发 ， 采 用 消息 和 actor 机 制 ， 而 不 是 像 Java 代 码 那 样 用 老 旧 的 锁 机 制 。 

这 些 不 是 Scala 的 全 部 内 容 ， 只 掌握 它们 也 不 可 能 让 你 变 成 Scala 开 发 高 手 。 它 们 是 用 来 吊 你 
胃口 的 ， 只 是 给 你 几 个 具体 示例 表明 Scala 可 能 适用 于 哪些 场合 。 要 走 得 更 远 , 就 得 做 更 深入 的 探 
索 。 你 可 以 找 些 在 线 资源 ， 也 可 以 找 本 完整 讲述 Scala 的 书 ， 比 如 Joshua Suereth 的 Scala in Depth 
( Manning, 2012 )。 

我 们 要 解释 的 第 一 个 特性 ， 也 是 Scala 跟 ]ava 最 重要 的 差别 ,就 是 它 语 法 上 的 精炼 性 ， 我们 就 
直 奔 主题 吧 。 


9.1.1 简约 的 Scala 


Scala 是 采用 静态 类 型 系统 的 编译 型 语言 。 也 就 是 说 Scala 代 码 应 该 和 Java 代 码 一 样 详细 。 可 
scala 俩 俩 很 精炼 ， 它 太 精 炼 卫 ， 看 起 来 简直 和 脚本 霹 言 一 样 。 因 此 Scala 开 发 人 员 更 加 快速 和 高 
效 ， 写 代码 的 速度 几乎 可 以 跟 用 动态 语言 编程 媲美 了 。 

我 们 来 看 一 些 非常 简单 的 代码 ， 了 解 一 下 Scala 的 构造 方法 和 类 。 比 如 要 写 一 个 简单 的 现金 流 
模型 类 。 直 要 用 户 提供 两 项 信息 : 现金 流 的 额度 和 货币 。 用 Scala 应 该 这 样 写 : 


claas CashFlowlamt : Double, curr : String) 1 
def amount() = amt 
def currencyl() ss Curr 


j 


这 个 类 只 有 四 行 (其 中 一 行 还 是 用 来 结束 的 右 括号 )。 不管 怎 样 ， 它 有 获取 方法 (但 没有 设 
置 方法 ) 作 为 参数 , 还 有 一 个 单 例 构 造 方 法 。 跟 Java 比 起 来 , 这 简直 太 划 算 了 ( 就 这 么 几 行 代码 )。 
请 看 相应 的 Java 人 代码 ; 


public class CashFlow { 
private final double amt; 
private final String curr; 


public CashFlow(double amt, String curr) | 


1 入 
三 


214 第 9 章 Scala: 简约 而 不 简单 


this.amt = 号 mn) 
this., curr = CUEI; 


| 


public double getAmt(}) 1 
return amt; 


public string getCurr{) | 
return curr:}i 
} 
} 


跟 Scala 相 比 ，Java 代 人 码 中 的 重复 信息 太 多 了 ， 就 是 这 种 重复 导致 了 Java 代 码 的 元 长 。 
选择 Scala， 让 开发 人 员 尽 量 减少 重复 信息 的 输入 ，IDE 的 界面 中 就 可 以 显示 更 多 内 容 。 面 对 
和 微 复 厅 点 的 逻辑 时 ， 开 发 人 员 就 能 见 到 更 多 代码 ， 因 此 也 有 望 能 掌握 理解 它 所 需 的 更 多 线索 。 


J 7 比 Java 版 址 75%。 据 估计 ， 一 行 代码 每 年 的 成 本 是 3 美元 。 
在 这 个 项 目的 生命 期 内 ， Scala 版 代码 的 维护 成 本 就 


既然 说 到 这 儿 了 ， 我 们 就 来 看 看 第 一 个 例子 中 展示 的 语法 点 。 

口 类 的 定义 ( 就 它 的 参数 而 言 ) 和 类 的 构造 方法 是 同一 个 东西 。Scala 中 可 以 有 其 他 的 “ 畏 
助 构造 方法 ”， 稍 后 就 会 谈 到 。 

口 类 默认 是 公开 的 ， 所 以 没 必要 加 上 public 关 键 字 。 

口 方法 的 退回 类 型 是 通过 类 型 推断 确定 的 , 但 要 在 定义 方法 的 def 从句 中 用 等 号 告诉 编译 器 
做 类 型 推断 。 

口 如 果 方 法 体 只 是 一 条 语句 (或 表达 式 )， 那 就 没 必要 用 大 括号 括 起 来 。 

口 Scala 不 像 Java 一 样 有 原始 类 型 。 数 字 类 型 也 是 对 象 。 

Scala 的 精炼 不 止 体现 在 这 些 方面 。 其 至 像 HelloWorld 这 样 简单 的 经 典 程 序 中 都 有 所 体现 ， 


object HelloWorld | 
def mainlargs : Array [String]}) 1 
val hello = "Helloe World!" 


println(lhello) 
} 
即便 在 这 个 最 基本 的 例子 中 ， 也 有 几 个 帮 我 们 去 除 套 路 化 代码 的 特性 。 
口 关键 字 objeect 告 诉 Scala 编 译 器 这 个 类 是 单 例 类 。 
口 调用 println() 没 必要 说 明 完 整 路 径 (感谢 默认 引入 )。 
口 没 必要 在 main () 方 法 前 指明 关键 字 public 和 static。 
口 不 必 声 明 hello 的 类 型 ， 编 译 器 会 自己 找 出 来 。 


只 自在 读书 @3 


www .Zizidiary .com 


9.1 走马 观 花 Scala 215 


口 不 必 声 明 main() 的 返回 类 型 ， 编 译 器 会 自动 设 为 Unit ( 等 价 于 Java 中 的 veid )。 

这 个 例子 中 还 有 些 相 关 语 法 需要 注意 一 下 。 

口 跟 Java 和 Groovy 不 一 样 ， 变 量 的 类 型 在 变量 名 之 后 。 

口 Scala 用 方 括 号 来 表示 证 型 ， 所 以 类 型 参数 的 表示 方法 是 Array[Sstringl ， 而 不 是 

String[]je 

口 Array 是 纯正 的 汉 型 。 

口 集合 类 型 必须 指明 泛 型 (不 能 像 Java 那 样 声 明生 类 型 ”), 

口 分 号 绝对 是 可 选 的 。 

口 val 就 相当 于 Java 中 的 final 变 量 ， 用 于 声明 一 个 不 可 变 变 量 。 

口 Scala 应 用 程序 的 初始 人 人口 总 是 在 object 中 。 

在 后 续 几 节 中 ,我 们 会 详细 解释 这 些 博 法 是 如 何 工 作 的 , 并 且 我 们 还 会 再 选 几 个 让 你 更 省 手 
指头 的 Scala 创 新 介绍 一 下 。 我 们 也 会 讨论 Scala 的 范 数 式 编程 ， 它 对 于 编写 精炼 的 代码 非常 有 帮 
助 。 现 在 ， 我 们 先 来 讨论 一 个 强大 的 Secala“ 本 地 ”特性 。 


9.1.2 match 表达 式 


Scala 有 一 种 非常 强大 的 结构 : match 表达 式 。 最 简单 的 match 用 法 跟 Java 的 switch 差 不 多 ， 
但 match 的 表达 力 要 强 得 多 , match 表 达 式 的 形式 取决 于 case 从 名 中 的 表达 式 结 构 。Scala 调 用 不 
同类 型 的 case 从 侣 模式 ， 但 要 注意 ， 这 些 所 谓 的 模式 跟 正 则 表达 式 里 的 “模式 ”是 截然 不 同 的 
( 尽管 在 match 表 达 式 里 也 可 以 用 正则 表达 式 模式 )。 

先 看 一 个 熟悉 的 例子 。1.3.1 节 那个 带 字符 串 的 swtich 被 翻译 成 了 Scala 代 码 ， 请 看 


var frenchDayofWeek = args(0) match | 


case "Sunday" => "Dimanche" 
case "Monday" -=> "Lundi" 


case "Tuesday" => "Mardi" 
case Wednesday" =»> "Mercredir 
case "Thursday”" => "Jeudi" 


case "Friday" =» "Vendredi" 
Case "Saturday”" =» "Samedi" 
Cage = "Error: "+ Args(0) +"' is not a day of the week" 


} 


printlnlfrenchDayOfWeek) 

我 们 在 这 个 例子 中 只 用 到 了 两 种 最 基本 的 模式 : 用 来 确定 是 周 几 的 常量 模式 和 处 理 默 认 情况 
的 _ 模 式 ， 后 面 我 们 还 会 遇 到 其 他 模式 。 

从 培 言 的 纯 雄 性 来 看 ， 可 以 说 Scala 的 语法 比 Java 更 清晰 ， 也 更 正规 ， 至 少 从 下 面 这 两 点 来 看 
是 这 样 的 : 


由 生 类 型 (raw type ) 是 指 不 带 类 型 参数 的 话 型 类 或 接口 。 比 如 证 型 类 Box<r>， 创 建 它 的 参数 化 类 型 时 要 指明 类 型 
参数 的 真实 类 丕 ; Box<Integer> intBox = new Box<>():。 如 果 和 忽略 了 类 型 参 数 ，Box rawBox = new Box1) ， 
则 是 创建 了 一 个 生 类 刑 。 一 一 译 者 注 
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默认 case 不 需要 男 外 一 个 不 同 的 关键 字 ; 
口 单个 case 不 会 像 Java 中 那样 进入 下 一 个 case， 所 以 也 不 需要 break。 
这 个 例子 中 的 其 他 语法 点 如 下 所 示 。 
口 关键 字 var 用 来 声明 一 个 可 变 ( 非 final ) 变量 。 没 有 必要 尽量 不 要 用 它 ， 但 有 时 候 确 实 
震 要 人 它 。 
口 数组 用 贺 括 号 访问 ， 比 如 args (0) 是 指 main1() 的 第 一 个 参数 。 
口 总 应 该 包括 融 认 case。 如 果 Scala 在 运行 时 在 所 有 case 中 都 找 不 到 匹配 项 ， 就 会 抛 出 
MatchError。 这 绝 不 是 你 想 看 到 的 。 
口 Scala 支 持 间 接 方 法 调用 ， 所 以 可 以 把 args (0) ,matchlf ,. .3}) 写 成 args 10) match 
x 
到 日 前 为 止 一 切 都 好 。match 看 起 来 就 像 稍 微 简洁 些 的 switch。 但 这 只 是 它 众多 模式 中 最 
像 Java 的 。Scala 中 有 大 量 使 用 不 同 模式 的 语言 结构 。 比 如 说 ， 有 一 种 类 型 化 模式 ， 对 于 处 理 类 型 
不 确定 的 数据 很 有 用 ， 不 用 像 Java 那 样 弄 一 堆 乱 烧 糟 的 类 型 转换 或 instancecof 测 试 ， 
def storagesizefobj: Any) = Dbj match I 
case 8: String =» 8.1length 
case i: TITnt 三 > 县 
CASE = -1 
| 
这 个 极其 简单 的 方法 以 一 个 any 类 型 ( 即 未 知 类 型 ) 的 值 为 参数 ,然后 用 模式 分 别处 理 String 
和 Int 类 型 的 值 。 每 个 case 都 给 要 处 理 的 值 绑 定 了 一 个 临时 别名 ， 以 便 必 要 时 可 以 调用 其 中 的 
方法 。 
在 Scala 的 异常 处 理 代 码 中 有 一 个 跟 变 量 模 式 非常 相似 的 语法 形式 。 下 面 是 一 段 改 编 自 第 1] 
关 ScalaTest 框 架 的 类 加 载 代码 ; 
def getReporter{repClassName: String, loader: ClassLoader); Reporter = | 
有 reportercl: java.lLang.Class[l ] = loader.loadCclass lrepClassName) 
reporterCl .newInastance.asIinatanceof [Reporter] 


catch 1 

case 已: ClassNotFoundException =» | 
val meg = "Can't load reporter clagas" 
Val iae = new IllegalhrgumentException (msg) 
iae.initCause(e) 
throw iae 

| 

case ee: InstantiationException => | 
val msg = "Can't instantiate Reporter" 
val iae = new IllegalArgumaent Exception (msg) 
1ae .initcCause le) 
throw iae 
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在 getReporter1) 中 ， 要 加 载 一 个 定制 的 report 类 (通过 反射 )， 以 便 在 运行 测试 集 时 输 
出 报告 ,在 类 加 载 和 实例 化 过 程 中 很 多 事 都 可 能 出 错 ,所 以 要 有 个 try-catch 块 来 保护 程序 执行 。 

catch 块 起 到 的 作用 就 跟 在 异常 类 型 上 放 match 表 达 式 类 似 。case 类 的 这 种 思路 还 可 以 进 一 
步 延 伸 ， 接 下 来 我 们 就 来 讨论 这 个 。 


9.1.3 case 类 


match 表 达 式 的 最 强 用 法 之 一 就 是 跟 case 类 ( 可 以 看 成 是 杖 举 概念 面向 对 象 的 扩展 ) 相 结合 。 
我 们 来 看 一 个 温度 过 高 发 出 报 塞 信号 的 例子 : 

case Class TemperatureAlarmltemp : Double) 
单 这 一 行 代码 就 可 以 定义 一 个 绝对 有 效 的 case 类 。 在 Java 中 相应 的 类 大 概 应 该 是 这 样子 : 
public class TemperatureAlarm 1 

private final double temp; 

public TemperatureAlarmldouble temp) | 

this.temp = temp:; 
| 


Publiec double 9etTemp 1() | 
return temp; 
| 


EOverride 
public String tosString() | 
return "TemperatureAlarm [temp=" + temp + "]"; 


| 


EOverride 
public int hashCode() 1 
final int prime = 31; 
int result = 工 ; 
long temp; 
temp = Double.doubleToLongBits (this.templ); 
result = prime * result + (int) {temp ” (temp >>> 32)); 
return result,; 


} 


BOverride 
public boolean equals (Object obj) 1 
if (this =s obj) 
return true; 
if (obj == null) 
return 1alse; 
if (getClass() != obj.getClass{)} 
return false:; 
TemperatureAlarm other = (TemperatureAlarm) ob]j; 
if {Double.doubleToLongBitasltemp) != 
Double.doubleToLongBitas lotheaer .temp)) 
return false; 
return true; 
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只 需 加 个 case 关 键 字 就 可 以 让 Scala 编 译 器 生成 这 些 额外 的 方法 。 它 还 会 生成 很 多 人 额外 的 架 
子 方法 。 大 多 数 情况 下 , 开发 人 员 都 不 会 直接 使 用 这 些 方法 。 它们 是 为 某 些 Scala 特 性 提供 运行 时 
支持 的 一 一 能 以 “自然 的 Scala” 方 式 使 用 case 类 。 

创建 case 类 实例 不 需要 关键 字 new， 像 这 样 : 

val alarm = TemperatureAlarm(l99.9) 

这 进一步 强化 了 case 类 是 类 似 于 “参数 化 枚 举 类 型 ”或 某 种 形式 的 值 类 型 的 观点 。 


站 本 i i A Ee ee 和 L EE | 蕊 Ee 和 = ee = 到 
es nl Wt i FT ee 9 村 ee. Ee! a | | pe 二 rl A 0.. sel, 


Pe , 中 丁子 出 
a i en i ue ET. be ea eee re rn + 
em ee ee ee ee = a ee ET nd a | ps 
er | + ee hh de en i Pt -ee a i : 1 - Ee Ee 直 二 Ee i 和 
E 5 mi rh -a i i ol Ky 党 总 Te mee a 二 -" [= A .re 5 CT mt pe 一 
Te ee 二 :各 性 2 让 一 一 中 二 和 [ 1 本 Ws 和 | ees Li 全 机 下 f 
要 与 i 站 玫 汉 “成 FE FE i 只 9 rr 
L | 由 证 1 
PP 和 中 | 了 和 
这 


err “引用 相等 " 是 个 错误 。 所 以 在 Scala 中 ，- Po re 
的 。 如 果 需 要 判断 引用 相等 ， 可 以 用 ===。case 类 的 . eat ht ein et 
参数 值 都 一 样 时 才 会 返回 true。 , : 人 


case 类 跟 构造 右 模 式 非常 合 ， 请 丰 : 


def ctorMatchExamplelsthg : AnyRef) = | 
val mag = sthg match 1 
Case Heartbeat => 0 
case TemperatureAlarm(ltemp) =»> "Tripped at temp "+ temp 
Case => "No match" 


| 


println (msg) 
我 们 去 看 看 Scala 观 光 之 旅 的 最 后 一 站 : 基于 actor 的 并 发 结构 。 
9.1.4 actor 


Scala 选 择 用 actor 机 人 制 来 实现 并 发 编程 。 它们 提供 了 一 个 异步 并 发 模型 , 通过 在 代码 单元 间 传 
递 消息 实现 并 发 。 很 多 开发 人 员 都 发 现 这 种 并 发 模型 比 Java 提 供 的 基于 锁 机 制 、 默 认 共 享 的 并 发 
异型 易 用 ( 不 过 Scala 的 底层 模型 也 是 JMM )。 

来 看 个 例子 。 假 设 我 们 在 第 4 章 遇 到 的 兽医 需要 监控 诊所 里 动物 的 健康 状况 ( 尤其 是 体温 )。 
按 我 们 的 想法 ， 温度 感 应 旬 应 该 会 将 它们 的 读数 消息 发 送 给 中 心 监控 软件 。 

在 Scala 中 ， 我 们 可 以 用 一 个 actor 类 TemperatureMonitor 对 这 种 设置 建 模 。 应 该 有 两 种 不 
同 的 消息 : 一 种 是 标准 的 “心跳 ”消息 ,一 种 是 TemperatureAlarm 消 息 。 es 一 个 
参数 ， 表 明 那 个 警报 融 的 温度 超出 了 限 值 。 代 码 清单 9-1 中 列 出 了 这 些 类 的 代码 。 


代码 清单 9-1 与 actor 的 简单 通信 
case object Heartbeat 
case clasgs TemperatureAlarmltemp : Double) 


import scala.actors, 


class TemperatureMonitor extends Actor | 


1 入 
三 


9.2 Scala 能 用 在 我 的 项 目 中 吗 219 


var tripped : Boolean = false 
var tripTemp : Double = 0.0 重 写 acter 中 的 act1) 方 法 
def actfl = 1 


while (true) 1 


receive | 一 
Case Heartbeat =»> 0 接受 新 消息 
case TemperatureAlarmltemp) => 


tripped = true 
tripTemp = temp 
cage => println("No match") 


} 

| 

监控 actor 会 对 三 种 不 同 的 case 做 出 响应 (通过 receive) 第 一 个 是 心跳 消息 ， 告 诉 你 一 切 
正常 。 因 为 这 个 case 类 没有 参数 ， 所 以 技术 上 来 说 它 是 一 个 单 例 实例 ， 可 以 按 case 对 象 引 用 。 
actor 在 收 到 心跳 消息 时 什么 也 不 用 做 。 

如 果 收 到 remperaturealarm 滑 上 电 ，actor 会 保存 索 报 此 上 的 温度 值 。 你 应 该 想象 得 出 ， 兽 
医 有 另外 的 代码 定期 检查 TemperatureMonitor actor， 看 有 没有 警报 被 触发 。 

最 后 还 有 个 aefault case。 这 是 为 了 确保 有 任何 不 期 而 至 的 消息 汶 进 actor 环 境 时 能 被 捕获 
到 。 如 果 没 有 这 个 一 切 全 包 的 case，actor 如 果 看 到 不 认识 的 消息 类 型 就 会 抛 出 异常 。 我 们 在 本 
章 的 最 后 还 会 再 次 讨论 actor 的 更 多 细节 , 但 Scala 的 并 发 是 个 非常 大 的 主题 , 而 且 在 这 本 书 里 我 们 
也 不 想 让 你 线 莹 瑟 止 。 

我 们 快速 浏览 了 Scala 的 一 些 亮点 ,希望 其 中 的 某 些 特性 已 经 燃 起 了 你 的 兴趣 之 火 , 在 下 一 节 ， 
我 们 会 论点 时 间 聊 聊 你 可 能 会 (也 可 能 不 会 ) 在 目 己 的 项 目 中 选择 使 用 Scala 的 原因 。 


9.2 ”Scala 能 用 在 我 的 项 目 中 吗 


在 Java 项 目 里 增加 一 门 博 言 总 该 有 正当 的 理由 和 根据 。 在 这 一 六 , 我 们 希望 你 想 想 那些 理由 ， 
以 及 如 何 把 它们 应 用 到 你 的 项 目 中 。 

一 开始 我 们 会 快速 比较 一 下 Scala 跟 Java， 然后 看 看 什么 时 候 ， 以 及 如 何 开 始 使 用 Scala。 为 了 
让 这 一 篇 幅 较 短 的 章节 更 完整 , 我 们 会 看 一 些 示 警 信和 号， 当 出 现 这 些 信和 号 时 ， Scala 可 能 不 是 最 适 
合 你 项 目的 语言 。 


9.2.1 Scala 和 Java 的 比较 


我 们 在 表 9-1 中 对 这 两 种 语言 的 主要 差异 做 了 汇总 。 语 言 的 “表皮 层 ” 是 指 该 语言 关键 字 的 
数量 和 开发 人 员 用 它 干 活 必须 掌握 的 独立 语言 结构 的 数量 。 

这 些 差异 是 Scala 得 到 Java 开 发 人 员 青 睐 , 可 以 把 它 当 做 某 些 项 目 或 组 件 的 备 选 语言 的 部 分 原 
因 。 接 下 来 我 们 会 给 出 把 $cala 引 人 项 目的 更 多 细节 。 
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表 9-1 比较 Scala 和 Java 

特 性 Java : Scala 
类 型 系统 静态 的 、 非 党 繁琐 静态 的 ， 但 使 用 了 大量 类 型 推断 
争 语言 金宇 塔 层级 稳定 稳定 、 动 态 
井 恬 模型 基于 镇 机 制 基于 actor 机 制 
国 数 式 编程 需要 严格 遵守 的 特殊 编码 方式 ， 不 自然 内 置 支持 ， 语 言 的 一 部 分 ( 纯 天 然 ) 
表皮 层 小 型 /中 型 大 型 /超大 型 
语法 风格 简单 、 常 规 、 比 较 第 开 灵活 、 精 炼 、 很 多 特殊 情况 


9.2.2 何 时 以 及 如 何 开 始 使 用 Scala 


我 们 在 第 7 划 讨 论 过 , 在 已 有 项 目 中 引入 新 语言 时 , 最 好 从 风险 比较 低 的 区 域 开 始 。ScalaTest 
测试 框架 ( 见 第 11 章 ) 就 是 这 样 的 低 风 险 区 。 如 果 Scala 实 验 不 顺利 ， 那 所 有 的 成 本 就 是 开发 人 员 
浪费 的 时 间 (这 些 单元 测试 有 可 能 变 成 普通 的 JUnit 测 试 )。 

一 般 来 说 ， 适 合 在 项 目 中 引 作 Scala 的 组 件 应 该 基本 满足 下 面 这 些 条 件 : 

口 你 有 信心 评估 所 需 的 工作 量 ; 

口 问题 域 边界 明确 ， 定 义 清晰 ; 

口 需求 说 明正 确 ; 

口 与 其 他 组 件 的 互 操作 性 需求 已 知 ; 

口 确定 了 愿意 学 习 新 语言 的 开发 人 员 。 

经 过 诬 思 熟 虑 选 定 合 适 区 域 后 , 你 就 可 以 开始 实现 自己 的 第 一 个 Scala 组 件 了 。 下 面 有 些 指 导 
原则 ， 事 实证 明 它 们 能 让 初始 组 件 按部就班 地 进行 ， 

口 以 快速 扣 杀 开始 ; 

口 尽早 跟 已 有 的 Java 组 件 测 试 交 互 操作 ，; 

口 有 定义 扣 杀 成 功 或 失败 的 入 门 标准 (基于 需求 ); 

口 如 果 扣 杀 失 败 ， 要 有 Bi 计划 ; 

口 在 预算 中 为 新 组 件 留 出 额外 的 重 构 时 间 ( 用 新 语言 编写 的 第 一 个 项 目 欠 下 的 技术 债 几乎 

肯定 会 比 用 团队 已 经 熟悉 的 语言 编 瑟 来 得 高 )。 

在 评 佑 Scala 时 ， 男 外 一 个 应 该 考虑 的 是 检查 那些 明显 让 Scala 对 项 目 来 说 不 太 理想 的 迹象 。 


9.2.3 ”Scala 可 能 不 适合 当前 项 目的 迹象 


下 面 这 些 迹象 表明 Scala 可 能 并 不 适合 你 的 项 目 。 如 果 出 现 了 其 中 一 个 或 多 个 迹象 , 你 应 该 慎 
重 考虑 引入 Scala 的 时 机 是 否 恰当 。 如 果 超过 两 个 ， 那 基本 就 没什么 戏 了 。 

D 受到 了 业务 小 组 和 其 他 程序 支持 小 组 的 抵制 ， 或 缺乏 动力 。 

O 开发 团队 没有 明显 的 学 习 Scala 的 动力 。 

小 组 中 分 帮 结 派 或 政治 上 存在 巨大 分 歧 。 
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口 小 组 中 高 级 技术 人 员 的 支持 力度 不 够 。 

口 截止 日 期 太 紧 张 ( 没 时 间 学 习 新 语言 )。 

另外 一 个 要 密切 关注 的 因素 是 ,团队 是 否 分 散在 全 球 各 地 。 如 果 你 用 来 开发 (或 广 持 ) Scala 
代码 的 员工 分 散在 几 个 地 方 ， 那 会 增加 Scala 培 训 人 人员 的 成 本 和 贷 担 。 

现在 我 们 已 经 讨论 过 把 Scala 引 人 项 目的 机 制 了 ， 接 下 来 该 去 看 看 Scala 的 语法 了 。 我 们 会 重 
点 关注 让 Java 开 发 人 员 更 轻松 的 特性 ， 辟 励 更 加 紧凑 的 代码 ， 少 点 套路 化 和 挥 之 不 去 的 柳 琐 。 
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我 们 在 这 一 节 会 先 介绍 一 下 Scala 编 译 器 和 交互 环境 ( REPL )。 然后 讨论 类 型 推断 , 接 者 是 方 
法 声明 ( 跟 你 所 熟悉 的 Java 方 式 不 太一 样 )。 这 两 个 特性 能 帮 你 减少 大 量 的 套路 化 代码 ， 从 而 提 
局 生产 力 。 

我 们 会 谈 到 Scala 的 代码 封包 方式 和 更 强大 的 import 语 句 ， 然 后 详细 讲解 一 下 Scala 中 的 循环 
和 控制 结构 。 这 些 特性 植 根 于 跟 Java 差 异 巨大 的 编程 传统 ， 所 以 我 们 会 借 此 机 会 讨论 一 下 Scala 
的 函数 式 编程 ， 包 括 函 数 式 的 循环 结构 、match 表 达 式 和 函数 字面 值 。 

看 过 这 些 之 后 ,本 章 剩 下 的 大 部 分 内 容 对 你 来 说 都 没什么 问题 了 , 你 可 以 自信 地 说 自己 有 能 
力 成 为 一 名 Scala 程 序 员 了 。 来 吧 ， 现 在 我 们 就 开始 讨论 编译 融和 内 置 的 交互 环境 。 


9.3.1 使 用 编译 器 和 REPL 


Scala 是 编 详 型 语言 ， 所 以 执行 Scala 程 序 通 常 要 把 它们 先 编 详 成 .class 文 件 ， 然 后 在 类 足 征 上 
有 scala-library.jar ( Scala 运 行 时 类 库 ) 的 JVM 环 境 中 执行 。 

如 果 你 还 设 装 Scala， 请 在 继续 阅读 之 前 参见 附录 C， 了 解 如 何 安装 Scala。 样 例 程序 ( 9.1.1 
中 的 HelloWorld ) 可 以 用 scalac HelloWor1ld.scala 继 译 ( 如 果 你 正好 在 HelloWorld.scala 文 件 
所 在 的 目录 中 )。 

-日 得 到 .class 文 件 , 就 可 以 用 命令 scala Helloworlda 执 行 它 了 。 这 个 命令 会 启动 带 着 Scala 

运行 时 环境 的 JVM， 然 后 进入 类 文件 指定 的 main 方 法 。 

除了 编译 和 运行 ，Scala 还 有 个 内 置 的 交互 环境 ， 有 点 像 第 8 章 讲 的 Groovy 控 制 台 。 但 不 像 
Groovy，Scala 是 在 命令 行 环境 里 实现 的 。 这 就 是 说 在 典型 的 Unix/Linux 环 境 ( Path 设 置 正确 ) 中 ， 
你 可 以 敲 人 scala， 它 就 会 在 终端 窗口 内 打开 ， 而 不 会 再 弹出 一 个 新 窗口 。 


注意 这 类 交互 环境 有 时 被 称 为 读 入 一 计算 一 输出 ( Read-Eval-Print ) 循环 ,或 简称 为 REPL。 
这 在 动态 语言 中 很 常见 。 在 REPL 环 境 中 ， 前面 输 入 的 那些 行 的 计算 结果 还 在 ， 在 后 面 的 
表达 式 和 计算 中 还 可 以 用 ,在 本 章 的 剩余 部 分 ,我 们 偶尔 会 用 REPL 环 境 来 演示 Scala 语 法 。 


现在 我 们 开始 讨论 下 一 个 大 特性 Scala 的 高 级 类 型 推断 。 
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9.3.2 ”类 型 推断 


在 读 前 面 的 代码 时 你 可 能 已 经 注意 到 了 ， 我 们 在 声明 变量 hello 为 val 时 ， 没 有 指明 它 是 什 
么 类 型 。 因 为 它 很 “明显 ”是 个 字符 串 。 表 面 上 来 看 这 有 点 像 Groovy， 变 量 没有 类 型 ( Groovy 
是 动态 类 型 语言 )， 但 其 实 Scala 代 码 中 所 发 生 的 事情 完全 不 同 。 

Scala 是 静态 类 型 语言 ( 所 以 变量 确实 有 明确 的 类 型 )， 但 它 的 编译 器 能 分 析 源 码 ， 并 且 一 般 
都 能 根据 上 下 文 推断 出 应 该 是 什么 类 型 。 如 果 Scala 自 己 能 确定 是 什么 类 型 , 就 不 用 你 亲自 告诉 它 了 。 

这 就 是 类 型 推断 , 我 们 已 经 提 过 好 几 次 了 。Scala 在 这 方面 的 能 力 非 常 突出 一 以 致 于 开发 人 员 
经 常 在 行云流水 一 样 的 代码 中 忘记 静态 类 型 。 这 经 常 让 Scala 更 有 动态 语言 的 “感觉 "。 


ee 四 一 人 了 
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不 是 值 的 类 型 ， 但 它 的 确 也 能 推 斯 值 的 类 型 。 


你 已 经 见 过 其 中 最 简单 的 例子 了 ， 关 键 字 var 和 val ，Scala 根 据 赋 给 变量 的 值 来 推断 它们 的 
类 型 。Scala 类 型 推断 的 另 一 个 重要 应 用 是 方法 声明 。 我 们 来 看 个 例子 ( Scala 的 AnvReE 就 是 Java 
中 的 Object ); 

def lenlobj : AnyRef) = | 

, obij .tostring.length 

这 是 一 个 类 型 推断 的 方法 。 通 过 检查 它 返 回 代码 中 的 java .1lang .string#1length 的 类 型 
(int )， 编 详 带 知道 这 个 方法 要 返回 Int 类 型 的 值 。 注 意 ， 这 个 方法 没有 显 式 指 定 返 回 类 型 ,我 
们 也 不 需要 用 return 关 键 字 。 实 际 上 ， 如 果 你 放 了 一 个 显 式 的 return 在 这 里 ， 像 这 样 ; 

def len(lobj : AnyRef) = 1 

return ob].tostring.length 

会 得 到 一 个 编译 时 错误 : 

error: method len has return Statement needs regsult type 

ts obj.tostring.length 

如 果 你 连 aef 中 的 = 也 省 略 了 ,编译 器 会 假定 这 个 方法 会 返回 Unit( 就 

除了 前 面 那些 限制 ， 还 有 两 个 类 型 推断 受 限 的 区 域 ; 

D 方法 声明 中 参数 的 类 型 一 一 传 给 方法 的 参数 必须 指定 类 型 ; 

口 闻 归 函数 一 一 Scala 编 译 带 不 能 推断 递归 耳 数 的 返回 类 型 。 

关于 Scala 的 方法 , 我 们 讨论 的 东西 已 经 不 少 了 , 但 还 算 不 上 系统 化 的 讨论 , 所 以 我 们 来 巩固 
一 下 你 已 经 学 过 的 东西 。 


昭 Java 里 返回 void 一 样 )。 


只 自 在 读书 @3 


WWwwW .Zizidiary .com 
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9.3.3 方法 


你 已 经 见 过 怎么 用 aef 关 键 字 定义 方法 了 。 随 着 你 对 Scala 越 来 越 熟悉 ,关于 Scala 的 方法 ,还 
有 些 你 应 该 知道 的 重要 事实 。 
口 Scala 没 有 static 关 键 字 , 跟 Java 中 的 static 方 法 对 应 的 方法 必须 放 在 Scala 的 opject( 单 
例 ) 结构 中 。 稍 后 我 们 会 向 你 介绍 相关 概念 ; 伴生 对 旬 
口 跟 Groovy (或 Clojure ) 相 比 ，Scala 诸 言 的 运行 时 要 重 得 多 。Scala 类 中 可 能 会 有 很 多 由 平 
人 台 目 动 生 成 的 额外 方法 。 
D 方法 调用 是 Scala 的 核心 概念 。 在 Scala 中 设 有 Java 中 那 种 意义 的 操作 符 。 
口 对 于 哪些 字符 可 以 出 现在 方法 的 名 称 中 ，Scala 比 Java 更 灵活 。 特 别 是 那些 在 其 他 语言 中 
作为 操作 符 的 字符 ， 在 Scala 中 可 能 是 合法 的 方法 名 【( 比如 加 号 + )。 
间接 方法 调用 ( 前 面 讲 过 ) 中 有 Scala 把 方法 调用 和 操作 符合 并 到 一 起 的 线索 。 举 个 例子 ， 比 
如 要 把 两 个 整 型 相 加 。 在 Java 中 ， 应 该 是 写 一 个 a+b 这 样 的 表达 式 。 在 Scala 中 你 也 可 以 这 样 写 ， 
但 不 止 这 样 ， 还 可 以 写成 a.+ (b)。 换 句 话 说， 你 调用 了 a 上 的 + () 方 法 ， 并 把 b 作 为 参数 传 给 它 。 
这 就 是 Scala 不 再 把 操作 符 当 做 一 个 独立 概念 的 秘密 。 


注意 ”你 可 能 已 经 注意 到 了 , a.+(b) 是 在 a 上 调用 方法 。 但 原始 类 型 的 变量 aa 怎么 会 有 方法 呢 ? 
9.4 节 会 给 出 完整 的 解释 。 但 现在 ， 你 只 要 知道 Scala 的 类 型 系统 认为 所 有 东西 都 是 对 象 ， 
所 以 你 可 以 在 任何 东西 上 调用 方法 ， 即 便 是 Java 里 的 原始 类 型 变量 也 行 。 


你 已 经 见 过 一 个 用 aef 关 键 字 声明 方法 的 例子 了 。 我 们 再 来 看 一 个 例子 ， 一 个 实现 阶乘 函数 
的 简单 好 归 方法 : 
def fact (base : Int} : int = 1 
it (base <= 0) 
retuwuren 1 
le 
return base * fact (lbase -= 1) 
} 
对 于 所 有 负数， 这 个 了 消 数 都 返回 !， 这 算是 作 商 吧 。 实 际 上 ， 负 数 的 阶乘 是 不 存在 的 ， 但 大 
家 都 是 朋友 嘛 。 它 看 起 来 有 点 像 Java: 有 返回 类 型 ( Int )， 并 用 return 关 键 宇 表明 把 哪个 值 交 
回 给 调用 者 。 唯 一 需要 注意 的 就 是 在 函数 体 代 码 块 定义 之 前 额外 符号 =。 
scala 中 还 有 另外 一 个 Java 中 设 有 概念 : 局 部 函数 。 它 是 在 男 外 一 个 函数 内 部 ( 并且 仅 在 这 一 
作用 域内 有 效 ) 定义 的 函数 。 如 果 开 发 人 员 想 要 一 个 辅助 函数 ， 又 不 想 把 实现 细节 暴露 给 外 部 ， 
这 是 一 个 简单 的 办 法 。 在 Java 中 除了 用 private 方 法 之 外 别 无 选择 ， 但 这 个 函数 对 于 同一 类 的 其 
他 方法 都 是 可 见 的 。 但 在 Scala 中 ， 你 只 要 这 样 写 就 行 了 : 


1 地 
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def fact2 (base : Int) : Int = | 


def factHelpertn : Int) : Int = 1 
return fact2(n-1) 


if (base == 站 |】 
reéeturn 1 
El] ae 
return base * factHelper (base) 
| 
factHelper () 在 fact2() 的 封闭 作用 域 之 外 绝对 是 不 可 见 的 。 
接 下 来 ， 我 们 去 看 看 Scala 如 何 处 理 代码 的 组 织 和 导 人 人 。 


9.3.4 导入 


Scala 对 包 的 使 用 跟 Java 一 样 ， 关 键 字 也 一 样 ， 分 别 是 package 和 import。Scala 可 以 毫 无 障 
碍 地 导 人 和 使 用 Java 的 包 和 类 。Scala 的 var 或 val 变 量 可 以 引用 任何 Java 类 的 实例 , 不 需要 任何 特 
殊 的 语法 或 处 理 : 

import Java.io.File 

import java.net. 

import scala.collection. {Map, Seq)} 

import java.util.{Date => UDate|} 


头 两 行 代码 跟 Java 里 的 标准 导入 和 通配符 导 人 人 一样。 第 三 行 用 一 行 导 入 一 个 包 里 的 多 个 类 。 
最 后 一 行 在 导 人 时 指定 了 类 的 别名 ( 避免 缩写 冲突 出 现 )。 

跟 Java 不 一 样 ，Scala 中 的 import 可 以 出 现在 代码 中 的 任何 位 置 (不仅 限于 文件 顶部 )， 这 样 
你 就 可 以 把 import 当 做 文件 的 一 部 分 分 离 出 来 。Scala 也 有 默认 导 人 ， 即 所 有 .scala 文 件 默认 都 会 
叶 人 scala._。 这 里 有 很 多 有 用 的 函数 , 包括 我 们 已 经 讨论 过 的 一 些 ， 比 如 println。 对 于 所 有 
默认 导 人 的 完整 细节 ， 请 参见 www.scala-lang.org/ 上 的 API 文 档 。 

我 们 接 下 来 讨论 怎么 控制 Scala 程 序 的 执行 流 。 这 可 能 和 你 熟悉 的 Java 跟 Groovy 有 些 差异 ，。 


9.3.5 ”循环 和 控制 结构 


Scala 在 控制 和 循环 结构 上 引信 了 几 个 有 点 绕 的 创新 。 在 我 
前 ， 先 来 看 几 个 老 朋友 ， 比 如 标准 的 while 循 环 : 
Var COUNter zz 1 
while (counter <= 10) { 
println("." * counter) 
COUNter 三 COUNter + 1 
} 
还 有 dao-while 形 式 ， 
Var Counter = 1 
do | 
println("." * counter) 
OUNter = COUNter + 1 
} while (counter <= 10) 


们 向 你 介绍 这 些 不 熟悉 的 形式 之 


1 于 
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另 一 个 是 基本 的 for 循 环 : 

for [i <- 1 to 10) printin(i) 

看 起 来 都 很 好 。 但 Scala 更 灵活 ， 比 如 条 件 Eor 循 环 : 
for 位 <- 1 to 10; if i 4$2 == 0) println(i) 

还 能 在 多 个 变量 上 循环 ， 比 如 : 


for (x <- 1 to 5S; YY <=- 1 to Xl 
println(" "“ % (xX = YY} + XtoString * ¥) 


这 些 多 出 来 的 形式 源 于 Scala 实 现 这 些 结构 的 根本 性 差异 。Scala 用 晴 数 式 编程 中 的 概念 ( 列 
表 推 导 式 ) 来 实现 for 循 环 。 

列表 推导 式 的 一 般 概念 是 对 一 个 列表 中 的 元 素 进行 转换 ( 或 过 滤 ， 比 如 在 用 条 件 for 循 环 
时 )。 这 会 产生 一 个 新 列表 ， 然 后 在 其 中 的 每 个 元 素 上 逐次 运行 for 循 环 体 中 的 代码 。 

甚至 把 要 过 滤 的 列表 和 for 代 码 块 分 开 都 是 有 可 能 的 ， 用 yield 关 键 字 。 比 如 下 面 这 段 
代码 : 

vAl 其 有 = for lx <- 2 to 11) vyield fact (x) 

for (factx <- Xs8) println(lfactx) 


这 段 代码 先 设置 新 集合 xs ,然后 用 第 二 个 for 循 环 逐 一 输出 其 中 的 值 。 如 果 你 需要 一 个 创建 
一 次 、 使 用 多 次 的 集合 ， 这 个 极其 好 用 。 

这 一 结构 能 成 立 是 因为 Scala 支 持 函 数 式 编程 ， 
思想 。 


9.3.6 ”Scala 的 函数 式 编 程 


我 们 在 7.5.2 节 提起 过 ，Scala 把 函数 当做 内 置 的 值 。 这 就 是 说 函数 可 以 放 进 var 或 val 中 ， 并 
和 其 他 任何 值 所 受 的 对 待 毫 无 二 致 。 这 被 称 为 函数 字面 值 (或 匿名 函数 )， 它 们 是 Scala 世 界 观 的 
重要 组 成 部 分 。 

在 Scala 中 写 国 数字 面值 非常 简单 。 甚 中 的 关键 是 箭头 => ，Scala 用 它 来 表示 取得 参数 列表 并 
传递 到 代码 块 中 : 

(< 动 数 泰 数 列表 >)】 => { ... 作为 代码 块 的 函数 体 .,。]】 

我 们 用 Scala 的 交互 环境 来 演示 一 下 。 下 面 这 个 例子 中 定义 的 函数 接受 一 个 Int 参 数 ， 人 然后 
琵 以 2， 


scala> Val doubler = (x : Int) => {2*x |} 
doubler:;: (Int) =s In = <functionls 


我 们 接 下 来 就 去 看 看 Scala 如 何 实现 函数 式 


scalas doubler (3) 
FEeEB4: Int = 在 


acala> doubler {#4) 
resS: Tnt = 8 


注意 看 Scala 怎 么 推 新 aoubler 的 类 型 。 它 的 类 型 是 “接受 一 个 Int 并 返回 TInt 的 函数 "。 这 样 
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的 类 型 用 Java 的 类 型 系统 还 不 能 以 令 人 完全 满意 的 方式 表示 。 你 看 ,调用 doubler 就 是 用 标准 的 
调用 语法 。 

我 们 把 这 个 概念 再 向 前 推进 一 点 。 在 Scala 中 ， 哨 数字 面值 只 是 值 。 并 且 是 沙 数 返回 的 值 。 这 
就 是 说 你 可 以 写 一 个 生产 函数 的 函数 一 一 接受 一 个 值 并 返回 一 个 新 的 函数 字面 值 。 

比如 说 ， 可 以 定义 一 个 命名 为 adder 的 卫 数 字 而 值 。adder () 能 生产 一 个 给 它们 的 参数 加 上 
一 个 常量 的 困 数 : 


scalas Val adder = (人 : Int) => { (x : Int) =»> x + nm | 
adder: {Int) => (INt) =»> Int = functionl> 


scalas val plus2 = adder (2) 
plus2:; (Int) =» Int = <functionls 


scalas plus2 (3) 
regs2: Int = 与 


scalas plus2 (4) 
res3: Tnt = 6 


看 到 了 吧 ，Scala 对 函数 字面 值 支持 得 很 好 。 实 际 上 ，Scala 代 码 一 般 都 能 用 非常 函数 的 世界 
观 来 编写 , 同时 也 能 用 更 加 命令 式 的 风格 编写 。 现在 我 们 所 做 的 不 过 是 刚刚 涉足 Scala 的 函数 式 编 
程 能 力 ， 但 重要 的 是 知道 它们 在 那里 。 

在 下 一 节 中 ， 我 们 会 讨论 Scala 的 对 象 模型 和 面向 对 象 方式 的 细节 。 在 一 些 重要 方面 ，Scala 
的 一 些 先 进 的 特性 使 得 它 对 面向 对 象 的 处 理 方式 跟 Java 差 异 很 大 。 


9.4 Scala 对 象 模型 : 相似 但 不 同 


Scala 有 时 被 称 为 “纯粹 ”的 面向 对 象 语 言 。 也 就 是 说 所 有 的 值 都 是 对 象 ， 所 以 面向 对 象 的 概 
念 几乎 随处 可 见 。 本 节 一 开始 ， 我 们 会 探索 一 下 “一 切 缘 对象” 的 后 果 。 这 个 主题 会 很 卓然 地 引 
导 我 们 去 思考 Scala 的 类 型 层级 。 

Scala 的 类 型 层级 跟 Java 有 几 个 重要 差异 , 包括 装 箱 和 拆 箱 等 Scala 处 理 原 始 类 型 的 方式 。 之 后 
我 们 会 考虑 Scala 的 构造 方法 和 类 定义 ， 以 及 它们 怎么 帮 你 少 写 代码 。 接 着 是 关于 trait ( 特质 ) 
的 话题 ， 然 后 再 讨论 Scala 的 单 例 、 伴 侣 和 包 对 象 。 本 刷 最 后 我 们 会 看 一 下 怎么 用 case 类 再 进 一 
步 减少 套路 化 代码 ， 并 以 一 个 有 警示 意义 的 Scala 语 法 故事 作为 结尾 。 

让 我 们 开始 吧 。 


9.4.1 一切 篆 对 象 


scala 的 观点 是 所 有 类 型 都 是 对 象 类 型 。 包 括 Java 所 谓 的 原始 类 型 。 图 9-1 展 示 了 Scala 的 类 型 
继承 关系 ， 包 括 所 有 值 类 型 ( 即 原始 类 型 ) 和 引用 类 型 ， 并 标注 了 与 Java 中 类 型 的 对 应 关系 。 
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图 9-1 ”Scala 中 的 继承 层级 


从 图 中 可 以 看 到 ，Unit 和 其 他 值 类 型 在 Scala 中 都 是 正确 的 类 型 。AnyRef 类 相当 于 java. 
lang .0bject。 每 次 见 到 AnyRef， 你 都 应 该 在 心里 把 它 换 成 Oobject。 它 之 所 以 没 叫 opject， 
因为 Scala 也 要 运行 在 .NET 运 行 时 平台 上 ， 所 以 它 要 给 这 个 概念 再 起 个 名 字 。 

Scala 用 extends 关 键 字 表示 类 的 继承 关系 ,而且 它 的 用 法 跟 Java 很 像 所 有 的 非 私 有 成 员 都 
会 被 继承 下 来 ， 两 种 类 型 之 间 也 会 建立 起 父 类 /于 类 的 关系 。 如 果 类 定义 中 没有 显 式 扩展 其 他 类 ， 
则 编译 器 会 认定 它 是 anyRef 的 直接 子 类 。 

一切 篆 对 象 ”的 原则 可 以 解释 使 用 中 组 符号 的 方法 调用 。9.3.3 节 中 的 obj ,meth (param) 
和 obj meth param 是 方法 调用 的 两 种 方式 ， 其 含义 是 一 样 的 。 现 在 你 应 该 明白 了 ，Java 中 的 表 
达 式 1+2 是 数值 原始 类 型 和 加 法 操作 符 的 表达 式 ， 而 Scala 中 与 之 对 应 的 1.+(2) 是 scala.Int 类 
上 的 方法 调用 。 

Scala 中 没有 因数 值 的 装 箱 操作 而 引起 的 困扰 ， 而 这 在 Java 里 很 常见 。 请 看 下 面 的 Java 代 码 : 


Integer one = new Integer (1); 
Integer Uno = new Integer(]1); 
System.out .println(one == uno),; 


你 可 能 觉得 很 奇怪 , 这 段 代码 的 输出 结果 居然 是 false。 而 Scala 中 对 数值 装 箱 及 相等 判断 的 
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方式 符合 我 们 的 常识 ， 这 有 以 下 几 个 好 人 处。 

口 数值 类 不 能 由 构造 方法 实例 化 。 它们 是 有 效 的 abstract 和 final 关 ( Java 中 不 允许 这 种 

组 合 )。 

口 得 到 数值 类 实例 的 唯一 办 法 就 是 作为 字面 值 。 这 能 确保 2 总 是 同一 个 2。 

口 判断 两 个 值 是 否 相等 所 用 的 == 方 法 的 定义 和 equals () 一 样 ， 不 是 引用 相等 。 

口 == 不 能 重 写 , 但 equals() 可 以 。 

口 对 于 引用 相等 的 判断 ，Scala 中 有 eag 方 法。 但 一 般 不 太 会 用 到 它 。 

现在 我 们 已 经 讨论 了 了 Scala 中 一 些 最 基本 的 面向 对 象 概 念 ， 还 需要 再 多 介绍 一 点 儿 Scala 的 语 
法 。 最 简单 的 就 是 Scala 的 构造 方法 。 


9.4.2 构造 方法 


Scala 的 类 必须 有 个 主 构造 方法 来 定义 该 类 所 需 的 参数 。 此 外 ,类 还 可 以 有 额外 的 辅助 构造 方 
法 。 这 些 辅助 构造 方法 都 用 this |) 表示 ， 但 它们 比 Java 的 重 载 构造 方法 限制 更 严格 。 

Scala 辅 助 构造 方法 的 第 一 条 语句 必须 调用 同一 个 类 中 的 男 一 个 构造 方法 (或 者 是 主 构造 方 
法 ,或 者 是 男 一 个 辅助 构造 方法 )。 这 种 限制 是 为 了 把 控制 流 引 导 到 主 构造 方法 上 ， 因 为 它 是 类 
的 唯一 其 正人 口 。 也 就 是 说 ， 辅 助 构造 方法 的 真实 作用 是 为 主 构造 方法 提供 默认 参数 。 

请 看 cashFlow 上 的 这 些 辅 助 构造 方法 : 


class CashFlow(amt : Double, curr : String) 1 
def thislamt : Double) = thist(lamt, "GBP") 
def this(lcurr : String) = this(0, curr) 


def amount = amt 
def currency = eurr 
} 
这 个 例子 中 有 个 辅助 函数 可 以 只 给 出 金额 ，cashFlow 会 假定 货币 是 英镑 。 另 一 个 辅助 函数 
可 以 只 给 出 货币 ,假定 金额 为 0。 
注意 我 们 定义 的 amount 和 currency 方 法 ， 都 没有 括号 或 参数 列表 ( 甚至 连 室 的 都 没有 )。 
这 是 告诉 编译 器 ， 在 调用 这 个 类 的 amount 和 currency 方 法 时 不 需要 括号 ， 像 这 样 : 
Val Wages = new CashFlow(2000.0) 


println'‘(wages.amount) 
println (wages,. currency) 


Scala 对 类 的 定义 基本 都 能 对 应 到 Java 中 。 但 在 面向 对 象 的 继承 方式 上 , Scala 所 采用 的 方式 跟 
Java 有 显 着 的 差异 。 下 一 人 就 来 讨论 它们 的 差异 。 


9.4.3 ”特质 
特质 是 Scala 面 问 对 象 网 程 方 式 的 主要 组 成 部 分 。 广 义 上 来 说 , 它们 和 Java 接 口 一 样 。 但 跟 Java 


接口 不 同 的 是 ， 特 质 中 可 以 给 出 方法 的 实现 ， 并 且 这 些 实现 可 以 由 具备 该 特质 的 不 同类 痊 享 。 
要 理解 它 所 解决 的 Java 回 题 ， 请 看 图 9-2 中 从 不 同 的 基础 类 继承 而 来 的 两 个 Java 类 。 如 果 这 两 
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个 类 都 要 具备 额外 的 相同 功能 ，Java 中 的 做 法 是 声明 它们 实现 了 相同 的 接口 。 


| 11101111 


| getNamel ) 
| i1011101010 四 
图 9-2 Java 模 型 中 的 实现 复制 

代码 清单 9-2 是 一 个 简单 的 Java 例 子 , 就 是 上 面 这 种 情况 的 代码 。 回忆 一 下 4.3.6 节 那个 和 兽医 诊 
所 的 例子 。 很 多 带 到 诊所 的 动物 都 会 被 植 人 芯片 , 以 便于 识别 。 比如 猫 和 狗 几 乎 肯定 会 这 么 处 理 ， 
但 其 他 物种 可 能 不 会 。 
镇 入 芯片 的 功能 需要 提取 到 单独 的 接口 中 。 我 们 来 修改 一 下 代码 清单 4-11 中 的 Java 代 码 ， 加 
人 这 一 功能 ( 为 了 让 代码 看 起 来 更 清晰 ， 我 们 省 略 了 examine () 方 法 ) 


代码 清单 9-2 ”说 明 实现 代码 的 复制 


public abstract class Pet | 
protected final String name' 


1011101000 


public Pet (String name ) | 
name = name ; 
} 
| 


public interface Chipped | 
string getName |); 


public class Cat extends Pet implements Chipped | 
public Cat (String name | 
super (name }); 


public String getName() | 
return name, 
| 
} 
public class Dog extends Pet implements Chipped | 
public Dog{(string name ) | 
Super (name ); 
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| 

public String getName() | 
return mame | 

| 


Dog 和 cat 中 都 有 同样 的 getName () 代码 ， 因 为 Java 接 口中 不 能 有 实现 代码 。 代 码 清单 9-3 是 
Scala 用 特质 实现 的 版 本 。 
代码 清单 9-3 用 Scala 实 现 的 宠物 类 


class Pet (name : String) 


trait Chipped | 
var chipName : String 
def getName = chipName 


class Cat (name : String) extends Pet (name : String) with Chipped I1 
var chipName = name 


class Doglname : String) extends Pet lname : String) with Chipped 1 
var chipNMame = name 
| 
Scala 要 求 在 于 类 中 必须 给 父 类 构造 方法 中 出 现 的 参数 赋值 。 但 在 特质 中 声明 的 方法 都 会 
被 子 类 继承 。 这 样 就 减少 了 重复 实现 。 你 看 ，cat 和 Dog 类 都 要 给 参数 name 赋 值 。 两 个 子 类 都 
可 以 访问 chippea 中 的 实现 一 一 在 此 例 中 ,参数 chipName 可 以 用 来 保存 写 在 芯片 上 的 宠物 的 
和 名字。 


9.4.4 单 例 和 伴生 对 象 


我 们 来 看 看 Scala 中 的 单 例 对 象 ( 即 用 关键 学 object 定 义 的 类 ) 是 如 何 实现 的 。 回 想 一 下 9.1.1 
中 的 HelloWorla: 
objaect HelloWorla | 
def main{args : Array [String]) 1 
val hello = "Hello World!" 
printlnlhello) 


| 

} 

如 果 这 是 Java， 你 会 觉得 这 段 代 码 应 该 变 成 一 个 HelloWorld.class 文 件 。 实 际 上 ，Scala 会 把 它 
编译 成 两 个 文件 : HelloWorld.class 和 HelloVWorldg.class。 

因为 这 就 是 普通 的 类 文件 , 所 以 你 可 以 用 第 5 曹 介 绍 的 反 编译 工具 javap 看 看 Scala 编 详 器 产生 
的 字 节 码 。 这 会 让 你 对 Scala 的 类 型 模型 及 其 实现 方式 有 更 多 的 了 解 。 代 码 清单 9-4 是 对 这 两 个 文 
件 运 行 javap -c -p 产 生 的 结果 
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Compiled from "HelloWorld.secala" 
publie final claass HelloWorld extends java.lang.Object | 
public static final void main(ljava.lang.Sstringl]):; 
Coae 
0: getstatic #11 


se // Field HelloWorlds .MODULES:LHelloWworldas; 
3: aload 0 | 取得 单 例 伴生 模块 


4: invokevirtual #13 


ss // Method HellowWorlds .main: ([Lijava/lang/Sstring;})V 
7: return | 调用 伴生 的 main() 方 法 
1 
Compiled from "HelloWorld.scala" 
public final class HelloWorlds extendse JjJava.lang.o0biject 
= implements scala.ScalaObject | 


public static final HelloWorlds$ MODULES:; | ] : 
publie static {}; | 单 例 伴生 实例 
Code: 
0: new 闪 写 :i class HelloWorlds 


3: invokespecial #12 // Method "<init>": ()YV 
6: return 


public void mainl(java.lang.Sstring[]),; 
Code: 


0: getetatic #19 // Field scala/Predef$ .MODULES$:Laecala/Predefs; 
a3: ldc #22 // String Helle World'! 


5: invokevirtual #26 
ms // Method scala/Predef$ .println: (Lijava/lang/Obiject;}Vv 


B88: return hw 
| 私有 构造 方法 
private HellowWorlds|(}.; 


Code: 
0; aload 0 
invokespecial #33 // Method java/lang/Object."<init>":(})V 
: aload 0 
: putetatic #35 // Field MODULES:LHelloworlds; 
: Yeturn 


wi 


} 

明白 “Scala 没 有 静态 方法 或 域 ” 这 话 是 从 何 而 来 的 了 吗 ” 除 了 这 些 结 构 ，Scala 编 译 器 还 自 
动 生成 了 单 例 模式 代码 ( 不 可 变 静 态 实例 和 私有 构造 方法 ), 并 把 它们 插 到 以 $ 结 尾 的 类 中 ,main1() 
方法 仍然 是 常规 的 实例 方法 ， 但 是 是 在 单 例 的 Helloworlas 类 实例 上 调用 的 。 

这 意味 着 在 这 一 对 .class 文 件 之 间 有 二 元 性 : 一 个 和 Scala 文 件 的 名 字 相 同 , 另外 一 个 加 了 个 $。 
静态 方法 和 域 被 放 在 了 第 二 个 单 例 类 中 。 

Scala 中 名 字 相 同 的 class 和 object 非 常常 见 。 在 这 种 情况 下 ， 单 例 类 被 当做 了 伴生 对 象 。 
Scala 源 文件 和 两 个 VM 类 ( 主 类 和 伴生 对 象 ) 之 间 的 关系 如 图 9-3 所 示 。 
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HelloWorld.scala HelloWorid 


100111001 周一 
OL0010101 


100111001 
pl O10010101 | 


HelloWorld$ 
图 9-3 Secala 单 例 对 得 
尽管 你 不 知道 ， 但 你 确实 已 经 遇 到 过 伴生 对 象 了 。 在 Helloworla 中 ， 你 没 必要 指定 
println() 方 法 在 哪个 类 中 , 它 看 起 来 像 个 静态 方法 ,所 以 你 应 该 能 想到 它 是 伴生 对 象 中 的 方法 。 
i 上 我 们 青 看 一 下 代码 清单 9-2 中 与 main1) 方 法 对 应 的 学 太 们 : 


public void mainljava.lang.String[]}; 


Code: 
0: getstatic #19 // Field scala/Predef$ .MODULES :Lecala/Predefs$; 
3: ldc #22 // String Hello World! 


S55: invokevirtual #26 
ms /| Method scala/Predef$ .println: (Liava, lang/Obiject;})Y 
8: return 


这 段 代 码 中 的 println() 及 其 他 随时 可 用 的 Scala 函 数 都 在 Scala .Predqef 类 的 伴生 对 象 中 ， 

伴生 对 象 在 其 相关 类 那里 有 特权 。 它 能 访问 该 类 的 私有 方法 。 这 使 得 Scala 能 以 合理 的 方式 定 
义 私有 辅助 构造 方法 。Scala 定 义 私有 构造 方法 的 语法 是 在 其 参数 列表 之 前 加 上 关键 字 private， 
穆 这 样 ， 

class CashFlow private (amt : Double, curr : String)} | 

} 

如 果 私有 的 构造 方法 是 主 方法 , 那 就 只 有 两 种 办 法 可 以 创建 该 类 的 实例 : 或 者 通过 伴生 对 象 
里 的 工厂 方法 (可 以 访问 私有 构造 方法 )， 或 者 调用 一 个 会 开 的 辅助 构造 方法 。 

接 下 来 我 们 要 进入 下 一 主题 ，Scala 的 case 类 。 你 已 经 遇 到 过 了 ,但 为 了 刷新 一 下 你 的 记忆 ， 
我 们 青 重复 一 次 它们 是 通过 自动 提供 一 些 基本 方法 来 减少 套路 化 代码 的 有 效 办 法 。 


9.4.5 case 类 和 match 表达 式 
我 们 用 Java 实 现 一 个 简单 的 实体 ， 比 如 Point 类 ， 如 代码 清单 9-$ 所 示 。 
代码 清单 9-5 一 -个 用 Java 实 现 的 简单 类 


public class Point 1 
private final int x; 
private final nt 站 ; 
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public Point (int x, int y) 1 

this,.x = 其 ; 

this.y = y; 套路 化 代码 
} 
public String tostring() 1{ 

return "Point (x: " + Xt+ "yi T+Y + ")"; 


Override 
public boolean equals (Object cbj) 1 
if (l(tobij inatanceof Point)) | 
return false; 
Point other = (Point)obj; 套路 化 代码 
return other.x == 其 && Other.y == 下; 


BOverride 
public int hashcode() | 
return 其 * 17 + Yi 


} 
| 


这 套路 化 代码 简直 太 多 了 ,而 且 更 精 的 是 , 像 hashCode()、tostring()、equals() 以 及 
所 有 的 获取 方法 通常 都 是 由 IDE 上 自动 生成 的 。 如 果 在 语言 内 校 的 内 部 完成 这 些 自 动 生 成 的 工作 ， 
用 更 简单 的 语法 电 不 是 更 好 ? 

scala 的 确 支 持 上 自动 生成 ，case 类 就 可 以 。 代 码 清单 9-5 可 以 非常 简单 ， 

case Claeses polntI(X : InRE，Y : Int) 

这 和 Java 那 段 长 长 的 代码 功能 一 样 ， 但 除了 更 短 ， 它 还 有 别 的 好 处 。 

比如 说 ， 用 Java 那 个 版 本 ， 如 果 要 修改 代码 ( 假设 要 加 个 z 坐 标 )， 就 必须 更 新 tostring () 
和 其 他 方法 。 实 际 上 ， 应 该 要 把 原来 那些 方法 全 部 删 掉 ， 然 后 让 IDE 再 重新 生成 一 次 。 

用 Scala 这 些 都 没 必要 , 因为 根本 就 没 显 式 定 义 需 要 跟着 更 新 的 方法 。 这 归结 为 一 个 非常 强 的 
理论 : 不 可 能 在 没 出 现 的 源码 中 弄 出 bug 来 。 

在 创建 新 的 case 类 实例 时 ， 关 键 字 new 可 以 省 略 。 代 码 可 以 写成 这 样 : 

val pythag = Point (3, 4) 

这 样 看 来 case 类 更 像 市 一 个 或 多 个 参数 的 枚 举 类 型 了 。 实 际 上 case 类 的 底层 实现 机 制 是 提 
供 一 个 创建 新 实例 的 工厂 方法 。 

我 们 来 看 一 下 case 类 的 主要 用 途 : 模式 和 match 表 达 式 。case 类 可 以 用 在 叫做 构造 器 
( Constructor ) 模式 的 Scala 模 式 类 型 里 ， 请 看 代码 清单 9-6。 


Val xaxig = Point (2, 0) 
val vaxis = Point (0, 3) 
val some = Point(5, 12) 
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val whereami = {p : Point) => p match | 
Case Point (XxX, 0) =»> "On the x-axia" 
caae Point (0, YY) =»> "On the y-axis" 
CASEe 三 3 "Out in the plane”" 


println (whereami (xaxis)) 
println(lwhereami (yaxis)) 
println (whereami (some)) 


我 们 在 9.6 节 讨论 actor 和 Scala 的 并 发 观点 时 会 再 次 拜访 Constructor 模 式 和 case 类 。 

在 结束 本 节 之 前 , 我 们 要 发 出 一 个 警告 。Scala 丰 富 的 语法 和 聪明 的 解析 器 能 够 用 一 些 非 常 精 
炼 和 优雅 的 办 法 来 表示 复杂 的 代码 。 但 Scala 没 有 正式 的 语言 规范 ， 并 且 新 特性 的 增加 非常 频繁 。 
你 应 该 多 加 小 心 一 一 即便 是 经 验 丰 富 的 Scala 码 家 有 时 也 会 被 语言 特性 出 其 不 意 的 表现 是 到 ,在 语 
法 特性 互相 结合 时 尤其 如 此 。 

我 们 来 看 一 个 例子 : 一 种 在 Scala 中 模拟 操作 符 重 载 的 办 法 。 


9.4.6 ”警世 寓言 


我 们 再 想 一 想 刚 刚 提 到 的 point case 类 。 你 可 能 想 要 用 一 种 简单 的 办 法 来 表示 坐标 的 相 加 ， 
或 者 坐标 的 线性 增长 。 如 果 你 数学 好 , 可 能 马上 就 会 意识 到 这 是 一 个 平面 坐标 上 的 向 量 空间 属性 。 
代码 清单 9-7 将 方法 定义 得 像 普 通 的 操作 符 一 样 。 


代码 清单 9-7 ”模拟 操作 符 重 载 
case Class Point(lx : Int，Y : Int) { 
def *(m : Int) = Point (this,.x * m, thig,.y * m) 
def 二 (other : Point) = Point (this.x + other.x, this.y + other.y) 


} 

var poin = Point(2, 3)} 
Var poina = FPoint (5, 7) 
println lpoin) 
printilntipoin 2) 
println(poin * 2) 
println(lpoin + poin2) 
运行 这 段 代 码 得 到 的 输出 应 该 是 ， 
Point (2,3) 

Polnt (5,7) 

Point (4,6) 

Point (7,10) 


这 下 应 该 能 看 出 Scala 的 case 类 跟 Java 里 的 等 价 物 相 比 有 多 好 了 吧 。 只 需要 很 少 的 代码 ， 就 
能 创造 出 一 个 很 友好 的 类 ， 产 生 合理 的 输出 。 定 义 + 和 * 方 法 后 ， 你 已 经 可 以 模拟 操作 符 重 载 了 。 
但 这 种 方式 有 问题 。 请 看 下 面 这 段 代 码 : 


var poin = Point (2, 3) 
printin(2 * poin) 


这 会 导致 编译 错误 


1 地 
OEEEE 


= 9.5 数据 结构 和 集合 235 


Error: OvVverloaded method value * with alternatives: 
{Double)Double <ands 
(Float)Float <and> 
(Long)Long <and> 
(Int) Int Bnds 
(Char) Int <aAnd> 
(Short) Int <and> 
(Byte) Int 

cannot be applied to (Point) 

printlnt{(2 * poin) 
看 


One error found 

尽管 在 case 类 Point 上 已 经 定义 了 方法 * (mm : Int)， 但 不 是 Scala 要 找 的 那个 方法 ， 所 以 出 
错 了 。 为 了 让 前 面 的 代码 编译 成 功 ， 需 要 在 Int 类 上 实现 * (p : Point) 方 法 。 这 是 不 可 能 的 ， 
所 以 操作 符 重 载 只 是 一 个 假象 。 

这 带 出 了 Scala 中 有 一 个 有 趣 的 问题 : 很 多 语法 特性 的 限制 在 某 些 情况 下 可 能 会 让 人 大 吃 一 
惊 。 Scala 的 语言 分 析 器 和 运行 时 环境 在 底层 做 了 大 量 工作 , 但 这 些 隐藏 的 机 制 是 建立 在 尽量 做 正 
确 的 事 的 基础 上 的 。 

我 们 对 Scala 面 向 对 象 实现 方式 的 介绍 到 这 里 就 结束 了 。 还 有 很 多 先进 特性 没 涉及 , 很 多 现代 
化 的 类 型 系统 和 对 象 思 想 在 Scala 中 都 有 实现 ， 所 以 如 果 感 兴趣 ，Scala 的 广 阐 天 地 对 你 来 说 大 有 
可 为 。 如 果 前 面 的 那些 内 容 色 起 了 你 对 Scala 的 类 型 系统 和 面 加 对象 实现 方式 的 兴趣 , 你 可 以 去 读 
一 读 Joshua Suereth 的 Scala in Depth (Manning，2012 )， 或 其 他 专门 介绍 Scala 的 图 书 。 

你 可 能 已 经 想到 了 , 这 些 语言 理论 应 用 的 一 个 重点 是 Scala 的 数据 结构 和 集合 , 这 也 是 我 们 下 
一 节 的 主要 内 容 。 


9.5 ”数据 结构 和 集合 


你 已 经 见 过 一 个 简单 的 Scala 数 据 结 构 List 了 。 它 在 任何 编程 语言 中 都 是 一 个 基本 的 数据 结 
构 ， 在 Scala 中 也 不 例外 。 我 们 会 花 点 时 间 探 究 一 下 List 的 细节 ， 然 后 去 研究 一 下 Map。 

接着 我 们 会 认真 研究 一 下 Scala 中 的 证 型 ， 包 括 与 Java 谤 型 的 差别 及 Java 沁 型 所 不 具备 的 能 
力 。 我 们 会 以 一 些 标准 的 Scala 集 合 为 例 展开 讨论 ， 以 便 让 你 了 解 其 原理 。 

先 从 Scala 集 合 的 几 个 一 般 性 原则 开始 , 特别 是 跟 它 的 不 可 变性 和 它 与 Java 集 合 的 交互 性 相关 
的 原则 。 


9.5.1 List 


Scala 中 集合 的 实现 方式 跟 Java 很 不 一 样 。 你 可 能 会 有 点 吃惊 ， 因 为 在 很 多 其 他 领域 ， 
部 在 重用 和 扩展 Java 的 组 件 、 概 念 。 

我 们 来 看 看 Scala 的 理念 所 带 来 的 最 大 差异 : 

口 Scala 集 合 通常 都 是 不 可 变 的 ; 

口 Scala 把 跟 列 表 类 似 的 集合 的 方方面面 分 解 成 了 不 同 的 概念 ; 
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口 Scala 构 建 Li st 核心 所 涉及 的 概念 非常 少 ; 

口 Scala 集 合 的 实现 方式 是 不 同类 型 的 集合 提供 的 用 户 体验 是 一 致 的 ; 

口 Scala 玻 励 开发 人 员 构建 自己 的 集合 类 ， 并 让 它们 用 起 来 像 内 置 的 集合 类 一 样 。 

我 们 会 逐一 讨论 这 些 差异 。 

1. 不 可 变 和 可 变 集 合 

你 首先 要 知道 ，Scala 的 集合 既 有 不 可 变 的 版 本 , 也 有 可 变 的 版 本 , 并 且 不 可 变 版 本 是 默认 的 
(所 有 Scala 源 文件 都 可 以 随时 访问 )。 

我 们 需要 分 辨 可 变 集 合 和 可 变 内 容 之 间 的 本 质 区 别 。 请 看 代码 清单 9-8。 


代码 清单 9-8 ”可 变 和 不 可 变 
import scala.collection.mutable .LinkedLiest 
import scala.collection.JavaConverasions. 
import JjJava.util.ArrayList 


bject ListExamplea | 
def main{(args : Array [String]) I 
var list = List'(1,2,3) 


1igst = Jigst :+ 4 
println(list) 

列表 追加 方 ; 
val linkligst = LinkedList(1,2,3) | 表 追 加 方法 


linkliast .append (LinkedList (4})) 
println(linklist) 


val jlist = new ArrayList [String] () 
jliat .add ("foo") 
val slist = jlist.toList 
: println(slist) 
| | 
如 上 所 示 ，1ist 的 引用 是 可 变 的 (是 var )。 它 指向 一 个 不 可 变 列 表 实 例 ， 所 以 可 以 通过 重 
新 赋值 指向 新 对 象 。:+ 方 法 返回 一 个 新 的 (不 可 变 ) List 实 例 ， 这 个 新 实例 中 含有 新 追加 的 
相反 ,1inklist 是 指向 一 个 LinkedList 的 不 可 变 引 用 (是 val ), 而 LinkedList 实 例 是 不 
可 变 的 。linklist 的 内 容 可 以 修改 ， 比 如 在 其 上 调用 appena() 。 这 种 区 别 如 图 9-4 所 示 。 
代码 清单 9-8 中 还 演示 了 一 组 转换 函数 : 用 来 对 Java 集 合 和 相应 的 Scala 集 合 进行 相互 转换 的 
JavaConversions 类 ，。 
2. List 的 特质 
Scala 选 择 强 调集 合 的 特质 和 行为 ， 这 是 它 与 众 不 同 的 男 一 个 重要 之 处 。 我 们 以 Java 的 
ArravyList 为 例 。 除 了 object， 这 个 类 还 直接 或 间接 地 扩展 了 ， 
器 java.util.AbstractList: 


加 java,uti1l .abstractColLlLection。 


1 于 
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Hot … 


Ist --- 


本 


linkist —» 1 : 2 ; 3 : 4 


和 和 i 曾 | 刷 | 硬 | 鹿 员 | 呈 呈 和 是 帮 | 中 嘱 别 中 和 


图 9-4 不 可 变 和 可 变 集合 
还 有 接口 ，ArrayList 或 它 的 某 个 父 类 实现 了 表 9-2 中 列 出 的 接口 。 
表 9-2 ArrayList 实 现 的 Java 接 口 


Serializable Cloneable Iterable 
Collect ion List RandomAccess 


对 于 Scala， 情 况 要 稍微 复杂 一 点 。 以 LinkedList 为 例 ， 与 它 提供 的 功能 相关 的 类 或 特质 多 
达 27 个 ， 如 表 9-3 所 示 。 
表 9-3 LinkedList 实 现 的 Scala 接 口 


Serializable LinkedLietLik LinearSedq 
LinearSegLike Cloneable Seq 

SeoqLike GensSedq GenSegqLike 
PartialFunction Functionl Iterable 
IterableLike Equals GenIlterable 
GenIterableLike Mutable Traversable 
GenTraversable GenTraversableTemplate TraversableLike 
GenTraversableLike Parallelizable TraversableOnce 


Scala 的 集合 类 彼此 之 间 的 差异 并 不 像 Java 那 么 明显 。 在 Java 中 ，List、Map、Set 等 ， 根 据 
使 用 时 的 具体 类 型 会 有 不 同 的 处理 模式 ,但 在 Scala 中 ,由 于 使 用 了 特质 ,类 型 的 细 化 程度 要 比 Java 
高 得 多 。 因 此 你 可 以 把 注意 力 放 在 集合 的 各 种 性 质 上 ， 使 用 更 加 贴近 需求 的 类 型 精确 表达 你 的 
意图 。 

因此 ，Scala 的 集合 处 理 代 码 要 比 Java 的 看 起 来 更 加 整齐 。 


内 自在 读书 他 
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如 你 所 料 ，Scala 既 支持 不 可 本 rm RS 。set 的 典型 用 法 器 Java 里 的 模式 
一 样 : 用 一 个 中 间 对 象 按 顺序 遍历 集合 中 的 元 素 。 但 Java 用 的 是 Tterator 或 Iterable, 而 Scal 
用 Traversable， 它 跟 Java 类 型 之 间 不 能 互 操作 。 


构建 列表 的 两 个 基础 是 : Nil 表 示 空 列表 ，: :操作 符 能 从 已 有 的 列表 构建 新 列表 。: :操作 符 
的 发 音 是 cons， 它 和 Clojure 的 (concat) 函数 ( 见 第 10 章 ) 还 有 关系 。 这 两 者 都 表明 Scala 植 根 于 
函数 式 编 程 一 一 最 终 可 以 追溯 到 Lisp 中 。 

cons 操 作 符 有 两 个 参数 :一 个 类 型 为 T 的 元 素 和 一 个 类 型 为 Dist [T] 的 对 象 。 它 会 把 两 个 参 
数 合 到 一 起 创建 一 个 新 的 List [T] 值 : 


Scalas Val x = 2 :+ 3 ::; Nil 
x: List [Tnt] = List (2, 3) 


男 外 ， 也 可 以 直接 这 样 写 ，; 


scalas val Xx = List (2, 3) 
Xx: Ligt [Int] = Ligt (2, 3) 


BCalas 1 :: 其 
res0: List [Int) = List(i, 2, 3) 


安 cons 操 作 符 的 定义 ,A :: B :: 的 信 是 有 上 的 它 的 意思 是 A : ee 
为 i eA :: B 是 类 型 为 List [{T] 的 值 ， i (A :: 
) i:: Oe 学 辽 汪 的 计算 机 和 学 家 会 说， 相关 性 的 。 
也 解释 了 为 什么 要 写成 2 : : Nil， 而 2 ;: 3 不 行 。: :的 

类 型 的 值 而 3 不 是 List。 


9.5.2 Map 
映射 也 是 一 种 经 典 的 数据 结构 。Java 最 常见 的 就 是 它 的 HashMap。 在 Scala 中 ， 不 可 变 的 Map 
类 是 默认 形态 ， 而 HashMap 是 标准 的 可 变形 态 
代码 清单 9-9 中 有 几 种 简单 、 标准 的 映射 定义 和 操作 。 
代码 清单 9-9 Scala 中 的 Map 


tt i collection.mutable.HashMap 


Var XxX = Mapll -> "hi", 2 -> "There") 
for ((key; vau) <- xXx} println(key + "; " + vau) 
X= XxX (3 -> "bye") 


val hm = HashMap(1 -> "hi", 2 -> "There") 
hm 中 至 (3 二 第 "bye") 
printlnt(hm) 


i 
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看 到 了 吧 ，Scala 定 义 映 射 字 面值 的 语法 简洁 可 爱 : Map (1 ->"hi"，2 -> "There")。 用 
简 头 符号 直观 地 表明 了 每 个 键 “ 指 向 ”的 值 。 要 从 映射 中 取 回 值 , 请 用 get () 方 法 ， 跟 Java 一 样 。 
可 变 和 不 可 变 映 射 都 用 + 表示 向 映射 中 添加 元 素 ( -表示 移 除 )。 但 这 个 有 些微 妙 ， 当 用 在 可 
变 映 射 上 时 ，+ 修 改 映 射 然后 返回 它 。 而 用 在 不 可 变 实例 上 时 ,返回 的 是 一 个 包含 新 的 键 / 值 对 的 
新 映射 。 这 会 导致 += 操 作 生 出现 以 下 边界 情况 : 
Scalas> Val m = Mapll -> "hi", 2 -> "There", 3 -> "bye", 4 -=> "quux") 


m: scala.collection.immutable.Mapl[lInt, java.lang.string] 
we = Mapll -=-» hi, 2 -> There, 3 -> bye, & -> guux) 


scala> Mm += (5 =»> "Blah"™) 
CONnBole>:10: error: reagsignment to val 
m += (5S -> "Blah") 


scala> Val hm = HashMap(l == "hi", 2 =»> "There", 3 = "bye", 4 == "gquux") 
hm: scala.collection .mutable.HashMap [Int, java.lang.string] 
= = Mapl3 => bye, 4 -> guux, 1 =-»> hi, 2 =»s There) 


scalas hm #= (5 -> "blah"') 
res6: hm .type = MaplS5 -> blah, 3 -> bye, 4 -> dUuUUX, 1 -> hi, 2 -> There) 


这 是 因为 += 在 不 可 变 和 可 变 映 射 中 的 实现 是 不 一 样 的 。 对 于 可 变 映 射 ，+= 是 一 个 方便 修改 
映射 的 方法 。 这 就 是 说 在 一 个 val 映 射 上 调用 这 个 方法 完全 合法 ( 就 像 Java 在 final HashMap 上 
调用 put () 一 样 ) 对 于 不 可 变 映射 ，+= 被 分 解 成 = 和 + 的 组 合 ， 就 像 在 代码 清单 9-9 里 一 样 。 它 不 
能 用 在 val 上 ， 因 为 val 不 允许 再 次 赋值 。 

代码 清单 9-9 中 还 有 一 个 不 错 的 语法 : for 循环。 这 用 到 了 列表 推导 式 〈( 见 9.3.5 节 ) 的 思想 ， 
但 结合 了 把 键 值 对 拆 分 成 键 和 值 的 做 法 。 这 称 为 对 解构 ， 是 Scala 中 男 一 个 继承 自 函 数 式 编程 的 
概念 。 

对 于 Scala 中 的 映射 和 它们 的 能 力 ， 我 们 仅仅 触及 了 冰山 一 角 ， 但 我 们 要 前 往 下 一 个 主题 了 : 
这 型 。 


9.5.3 ” 泛 型 


你 已 经 知道 ，Scala 用 方 括号 表示 参数 化 类 型 ， 而 且 你 也 已 经 见 过 一 些 基 本 的 Scala 数 据 结 
爸 了 。 我 们 继续 深信 ， 看 看 Scala 对 证 型 的 处 理 方式 跟 Java 有 什么 不 同 。 

首先 ， 如 果 在 定义 函数 的 参数 类 型 时 忽略 掉 了 泛 型 ， 看 看 会 发 生 什 么 : 

scala> def Junklx : List) = println("hi®) 

CONSBOle>:;:5:; error; type Ligst takes type parameters 

def junklx : List) = println("his) 

在 Java 中 ， 这 是 完全 合法 的 。 编 译 器 可 能 会 抱怨 ， 但 不 会 报错 。 而 在 Scala 中 ， 这 是 一 个 编译 
时 和 错误。 列表 ( 和 其 他 泛 型 ) 必须 参数 化 一 一 故事 讲 完了 ，Scala 没 有 Java“ 生 类 型 ”的 概念 。 

1. 泛 型 的 类 型 推断 

把 泛 型 赋值 给 一 个 变量 时 ，Scala 会 对 类 型 参数 做 出 恰当 的 类 型 推断 。 这 符合 Scala 一 贯 坚持 
的 类 型 推断 和 尽 可 能 去 掉 套 足 化 代码 的 风格 : 


1 地 
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Bcalas val XxX = List(1, 2, 3) 

x: Ligt [Int] = List (1, 2, 3) 

Scala 泛 型 中 有 个 特性 乍 一 看 可 能 觉得 奇怪 ,我 们 用 : : :操作 符 演示 一 下 ,看 到 下 面 两 个 列表 
联接 起 来 产生 了 新 的 列表 ， 你 就 明白 为 什么 说 它 奇 怪 卫 : 

scala> Val Y = Liast("cat", "dog", "bird") 

Y: List[java.lang.String)] = List(cat, dog, birad) 

eg = Ligt(l, 2, 3, Cat, dog, bird) 

奇怪 吧 ， 这 样 居 然 都 不 报错 ， 还 产生 了 新 的 List。 运 行 时 产生 了 一 个 Int 和 String 的 最 小 
公 父 类 (any ) 的 列表 。 

2. 江 型 示例 : 候诊 的 宠物 

假设 有 些 宠物 在 等 着 看 兽医 ， 而 你 要 建立 候诊 室 里 排队 队列 的 模型 。 代 码 清 单 9-10 是 个 不 错 
的 起 点 ， 用 的 是 一 些 你 已 经 熟悉 的 基础 类 和 辅助 阴 数 。 


代码 清单 9-10 ”候诊 的 宠物 
class Pet [name : String) 
class Cat (name : String) extends Pet lIname : String) 
class Dog lname : String) extends Pet (name : StrFing) 
class BengalKitrten(name : String) extends Cat (name : String) 


class Queue[T] (elts : T*) i1 


var elems = List[lT] (elts : 二 | , 
def enqueue (elem : T) = elems :1:: | 需要 类 型 提示 


def dequeue = | 
val resgult = élems.head 
elems = elems.tail 
result 


| 
} 


def examine (lq : Queue [Cat]) I 
println(l"Examining: ”+ gq.dequeue) 
} 
我 们 来 考虑 一 下 在 Scala 提 示 符 中 怎么 使 用 这 些 类 。 这 些 是 最 简单 的 例子 : 
acalas examine (new Queue (new Cat ("tiddlea"))) 
Examining: lineSs$object$$iwisiwsCatafbO0defe 


Bcala> examine (new Queue (new Pet ("george"))})) 
CONnscoles:10: error: type mismatceh,; 
found : FPet 
reguired;: Cat 
examine (new Queue (new Pet ("george"))}) 


到 目前 为 止 都 很 像 Java。 我 们 再 多 做 几 个 简单 的 例子 : 
scala> examine (new Queue (new BengalKittent("michael")}) 


Examining: line7$object$$9iwddiwdBengalKitrten®d6daald9a 


scalas Vvar kitties = new Queue (new BengalKittenl("michael")) 
kitties: Queue [BengalKitten)] = Queue®2976c6e4 


只 自 在 读书 @3 
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acala> examine (kitties) 
CONgsole>:;12: error: type mismatch; 
found : Queue [BengalKitten)] 
required: Queue [Cat] 
EL 
这 也 相当 平常 。 第 一 个 例子 没有 将 kitties 作 为 临时 变量 , Scala 的 类 型 推断 把 队列 的 类 型 作 
为 oueue [cat] ， 并 接受 和 了 michael 的 加 人 ， 因 为 它 的 类 型 是 cat 的 子 类 BengalKitten。 第 二 
个 例子 中 , 创建 了 变量 kitties， 显 式 声明 了 其 类 型 。 也 就 是 说 Scala 不 能 用 类 型 推断 ， 所 以 不 能 
接受 类 型 不 匹配 的 参数 。 
接 下 来 我 们 去 看 看 如 何 用 类 型 系统 的 类 型 变 体 解决 这 些 类 型 问题 , 特别 是 协 变 ( 类 型 变 体 还 
有 其 他 形态 ， 但 协 变 最 常用 )。 在 Java 中 ， 这 非常 灵活 ， 但 也 有 点 神秘 。Scala 和 Java 的 做 法 我 们 
都会 读 示 一 下 。 
3. 协 变 
“在 Java 中 ，List<String> 是 List<object> 的 子 类 吗 ? ”如 果 你 问 过 类 似 问 题 
题 就 是 为 你 准备 的 。 
默认 情况 下，Java 对 这 个 问题 的 回答 是 “不 是 "， 但 你 可 以 让 它 变 成 “是 "。 要 知道 怎么 做 ， 
请 看 下 面 的 代码 : 


public class MYyListe<T> I 
private List<T> 七 heLlet 


[， 那 这 个 话 


MyLiat<Cat> katzchen = new MyList<Cat>|(); 
MyList=? extends Pet> petExt = petl; 


? extends Pet 从 名 表示 petExt 是 一 个 部 分 未 知 的 类 型 参数 (Java 类 型 中 的 ? 读 作 “未 知 ”)。 
可 以 确定 的 是 MyList 的 类 型 参数 必须 是 Pet 或 Pet 的 子 类 。 这 样 在 将 类 型 参数 为 其 于 类 的 值 赋 给 
petExt 时 ，Java 编 译 器 就 不 会 阻拦 。 

这 就 相当 于 把 MyvDist<cat> 变 成 了 MyList<? extends Pet> 的 子 类 。 注 意 ， 这 种 子 类 关 
系 是 在 使 用 MyList 类 型 时 建立 起 来 的 ， 而 不 是 定义 时 。 类 型 的 这 个 特性 称 为 协 变 。 

Scala 的 做 法 跟 Java 不 同 。 它 不 是 在 使 用 类 型 时 定义 类 型 变 体 ,而 是 在 类 型 声明 时 显 式 指定 协 
变 。 这 样 做 有 几 个 优势 : 

口 编译 只 可 以 在 编 至 时 检查 不 竺 合 协 变 的 使 用 ; 

口 所 有 概念 上 的 思虑 都 交 给 了 类 型 编写 者 ， 而 不 是 抛 给 类 型 的 使 用 者 ; 

口 这 样 可 以 在 基础 集合 类 型 间 植 人 直观 的 关系 。 

理论 上 来 说 , 这 样 的确 不 如 Java 那 样 使 用 现场 的 变 体 更 灵活 , 但 在 实际 应 用 中 ，Scala 采 取 的 
方式 所 带 来 的 好 处 完全 可 以 抵消 这 种 不 便 。 大 多 数 程序 员 很 少 会 使 用 Java 泛 型 中 那些 真正 先进 的 
特性 。 

Scala 的 标准 集合 ， 比如 Dist，, 都 实现 了 协 变 ,这 就 是 说 List [BengalkKkitten] 是 List [Catl] 
的 了 类， 而 它 又 是 List [Pet] 的 子 类 。 我 们 来 实际 操练 一 下 ， 请 启动 解释 器 : 
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scala> val kits = new BengalKittenl"michael") :: Nil 
kits: List[BengalKitten] = Ligst (BengalKitten®7ledS401) 


scala> var katzen : List [Cat] = kits 
katzen: List[lCat] = List (BengalKitten®?7?ledS5401) 


gcalas Var hauatieren : List [Pet] = katzen 

haustieren: Ligst [Pet] = Ligst (BengalKitten®7TledSsa401) 

我 们 在 var 上 显 式 声明 了 类 型 ， 以 免 Scala 把 类 型 推断 得 过 罕 。 

对 Scala 沁 型 的 简略 探讨 到 这 里 就 结束 了 。 下 一 个 大 主题 是 Scala 在 并 发 实现 方式 上 的 创新 : 
放弃 了 多 线程 显 式 管 理 的 方式 ， 而 选用 了 actor 模 型 。 


9.6 actor 介绍 


Java 的 显 式 锁 和 同 世 模型 刻下 了 岁月 的 痕迹 。 在 最 初 设计 Java 语 言 时 , 它 是 一 个 奇妙 的 创新 ， 
但 也 埋 下 了 祸根 。Java 并 发 模型 本 质 上 是 面 对 两 难 境地 时 采取 折 中 策略 的 产物 。 

锁 太 少 ， 会 导致 并 发 代码 不 安全 ， 出 现 竞 态 条 件 。 锁 太 和 多， 系统 会 者 失 活 力 ， 代 码 瘫 闪 ， 工 
作 毫 无 进展 。 这 就 是 我 们 在 第 4 章 讨 论 过 的 ， 安 全 性 与 系统 活力 之 间 的 矛盾 。 

使 用 基于 锁 的 模型 ， 必 须 照 顾 到 给 定时 间 内 所 有 可 能 发 生 的 并 发 操作 。 但 随 着 程序 变 得 越 来 
越 大 ， 要 做 到 滴水 不 漏 会 变 得 越 来 越 困 难 。 尽 管 Java 有 办 法 缓解 一 些 问 题 ， 但 核心 问题 还 在 ， 如 
果 Javai 语 言 不 能 发 布 一 个 拒绝 向 后 非 容 的 版 本 ， 就 不 可 能 从 根本 上 解决 这 个 问题 。 

非 Java 语 言 有 机 会 从 头 开 始 。 备 选 语 言 可 以 不 向 程序 员 暴 露 锁 和 线程 的 底层 细节 ， 而 是 在 自 
己 的 运行 时 环境 中 提供 额外 的 并 发 支持 。 

这 应 该 没什么 好 奇怪 的 。 毕 竞 在 Java 刚 刚 出 现时 , Java 内 存 模型 就 受到 过 质疑 。 当 时 很 多 C 和 C++ 
开发 人 员 都 对 这 种 想法 感到 这 异 ， 怎 么 能 由 运行 时 负责 管理 内 存 ， 而 让 开发 人 员 远 离 这 些 细节 呢 ? 

我 们 来 看 一 下 Scala 基 于 actor 技 术 的 并 发 模型 ， 看 它 如 何 让 并 发 编程 变 了 样 (也 更 简单 )。 


9.6.1 代码 大 舞台 


actor 是 扩展 scala.actors.Actor, 并 实现 Jact() 方 法 的 对 象 , 希望 这 个 定义 能 跟 你 脑海 
中 对 Java 线 程 的 定义 相 呼 应 。 它们 最 大 的 差别 就 是 actor 在 大 多 数 情况 下 都 不 会 通过 共享 的 数据 进 
行 沟通 。 

程序 员 在 共享 数据 时 必须 采用 最 佳 实践 。 如果 你 想 在 actor 间 共享 状态 ，Scala 不 会 阻止 你 ,我 
们 只 是 认为 这 么 做 不 好 。actor 有 沟通 的 渠道 mailbox ， 从 另 一 个 上 下 文中 发 送 过 来 的 消息 〈 工 
作 项 ) 可 以 放 在 mailbox 中 交 给 actor， 请 参见 图 9-5。 


图 9-5 scala 的 actor 和 mailbox 
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要 创建 actor， 扩展 Actor 类 就 行 : 
import ecala.actors, 
class MyActor extends Actor 1 
def act(}) I 
up 
} 
这 看 起 来 跟 Java 代 码 中 声明 Threaa 的 子 类 很 像 。 跟 线程 一 样 ， 我 们 也 要 告诉 actor 开 始 启动 ， 
并 进入 请 县 接收 的 状态 ， 这 要 调用 start 1() 方 法 。 
Scala 同 样 提 供 了 创建 actor 的 工厂 方法 actor (与 Java 里 创建 eunnable 匿 名 实现 类 的 静态 工 
三 方法 相对 应 )。 用 它 写 出 来 的 Scala 代 码 很 精炼 : 


val myactor = actor { 


| 

传 给 actor 的 代码 块 会 变 成 act ( ) 方 法 中 的 内 容 。 另 外 ， 这 样 创建 的 actor 不 需要 再 单独 调用 
start()， 它 会 目 动 局 动 。 

这 是 一 块 香 甜 的 语法 糖 , 但 我 们 还 要 介绍 Scala 并 发 模型 的 核心 部 件 mailbox ， 所 以 别 回 味 了 ， 
现在 就 去 看 看 吧 。 


9.6.2 用 mailbox 跟 actor 通信 


从 为 一 个 对 象 给 actor 发 消息 很 简单 ， 只 要 在 actor 对 象 上 调用 ! 方 法 就 行 了 。 
然而 在 接收 端 要 有 代码 处 理 这 些 消 息 ， 否 则 它们 就 会 堆 在 mailbox 里 。 另 外 ，actor 方 法 体 通 
常 需要 有 个 循环 ， 以 便 能 处 理 所 有 流 人 的 消息 。 我 们 在 Scala REPL 中 实际 操练 一 下 : 
scala> import scala.actors.Actor. 
val myact = acEor | 
while (true) | 
pe => printilnl("I got mail:; "+ incoming) 
| 


| 
} 


myact: scala.actors.Actor = scala.actors.Actorssanons$le@a7sO0bbo 


scala> myact ! "Hello!™" 
I got mail: Hello! 


scala> myact ! "Goodbye!" 
I got mail: Goodbye! 


scala> myact |! 34 
I got mail: 34 


上 面 代码 中 的 receive 方 法 就 是 actor 对 消息 的 处 理 。 而 工厂 方法 的 参数 ( 代码 块 ) 则 是 消息 
处 理 方法 的 主体 。 
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注意 ”总 体 来 说 ，Scala 模 型 跟 我 们 第 4 章 (代码 清单 4-13 ) 讨论 的 处 理 模 式 相 似 ，Java 处 理 线程 
相当 于 acctor 的 角色 ，LinkedBlockingoueue 和 相当 于 Scala 中 的 mailbox 。Scala 只 是 以 非 
常 直 和 白 的 方式 为 这 种 模式 提供 了 语言 和 类 库 层 面 的 支持 ， 可 以 大 量 减 少 使 用 这 种 模式 时 
所 要 编写 的 套路 化 代码 。 


尽管 这 个 例子 非常 简单 ， 但 也 包 舍 了 很 多 使 用 actor 的 基础 知识 ; 

口 在 actor 方 法 中 要 用 循环 的 方式 处 理 接收 消息 流 ; 

口 用 receive 方 法 处 理 接收 到 的 消息 : 

口 用 一 组 case 作 为 receive 的 主体 。 

最 后 这 点 还 得 继续 讨论 。 这 一 组 case 被 称 为 偏 函数 "之 所 以 要 这 样 用 ,是 因为 Scala 中 的 actor 
还 有 一 点 比 Java 方 便 。 有 具体 来 说 就 是 mailbox 是 不 区 分 类 型 的 。 也 就 是 说 你 可 以 向 actor 发 送 任何 类 
型 的 消息 ，actor 可 以 用 类 型 化 模式 和 构造 器 模式 接收 不 同类 型 的 消息 。 

陈 了 这 些 基础 知识 ,这 里 还 有 一 些 使 用 actor 的 最 佳 实践 .编写 代码 应 该 斥 量 遵循 下 面 几 条 规则 

口 把 传人 消息 做 成 不 可 变 的 ; 

口 考虑 把 消息 类 型 做 成 case 类 ; 

口 不 要 在 actor 内 部 做 阻塞 操作 ， 一 个 也 别 做 。 

不 是 每 一 个 程 订 都 需要 遵守 所 有 的 最 住 实践 ,但 大 多 数 应 用 程序 应 该 都 能 从 这 些 建议 中 受益 。 

对 于 更 加 复 染 的 actor, 经 常 有 必要 控制 它 的 启动 和 关闭 。 关 闭 actor 通 常 都 是 用 带 有 Boolean 
条 件 判断 的 循环 。 如 果 你 至 欢 ， 也 可 以 将 actor 写 成 孙 数 式 的 风格 ,这样 传人 的 消息 就 不 会 影响 它 
的 状态 。 

Scala 对 基于 actor 的 并 发 编程 提供 的 支持 还 有 很 多 。 我 们 在 这 里 看 到 的 只 是 皮毛 。 如 果 想 全 面 
了 解 ， 请 参阅 Nilanjan Raychaudhuri 的 大 作 Scala in Action ( Manning, 2010 )。 


9.7 小结 


Scala 跟 Java 有 显著 的 差异 : 

口 文 持 更 灵活 的 函数 式 编程 风 档 ; 

口 类 型 推断 使 静态 语言 用 起 来 有 动态 语言 的 感觉 ; 

口 Scala 先 进 的 类 型 系统 扩展 了 Java 的 面向 对 象 概 念 。 

下 一 草 会 介绍 最 后 一 门 非 Java 语 言 : Lisp 方 言 Clojure， 这 可 能 是 从 各 方面 来 看 都 最 不 像 Java 
的 二 言 。 我 们 会 以 不 可 变性 、 国 数 式 编程 和 另 一 种 并 发 为 基础 展开 讨论 , 并 展示 Clojure 如 何 利用 
这 些 思想 构建 起 了 一 个 强大 无 比 、 美 丽 蜡 常 的 编程 环境 。 


山 在 Scala 中 ， 侦 耳 数 是 指 类 型 为 PartialFunction[-A,+B8] 的 鹃 数 。A 是 其 接受 的 函数 类 型 ，B 是 其 返回 的 结果 类 
型 。 贪 晒 数 最 大 的 特点 就 是 它 只 接受 其 参数 定义 域 的 一 个 子 集 ， 而 对 于 这 个 子 集 之 外 的 参数 则 殷 出 运行 时 异常 。 
这 与 case 滞 名 非常 契合 ， 因 为 我 们 在 使 用 case 语 句 时 常常 是 匹配 一 组 具体 的 模式 ， 最 后 用 “_ ”来 代表 币 余 的 模 
式 。 如 果 一 组 case 语 句 设 有 涵盖 所 有 的 情况 ， 那 么 这 组 case 语 句 就 可 以 被 看 做 是 一 个 偏 栅 数 ， 一 一 译 者 注 
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本 章 内 容 

D Clojure 实 体 和 状态 的 概念 

口 Clojure 的 REPL 

D Clojure 语 法 、 数 据 结 攀 和 序列 
口 Clojure 与 Java 的 交互 能 力 

口 Clojure 的 多 线程 开发 

口 软件 事务 内 存 


Clojure 跟 Java 以 及 我 们 前 面 研究 的 语言 差别 很 大 。Clojure 是 在 JVM 上 重新 实现 的 Lisp。Lisp 
是 最 古老 的 编程 语言 ， 如 果 你 对 它 还 不 熟悉 ， 设 关系 。 与 Lisp 语 言 家 族 有 关 的 一 切 ， 只 要 是 你 需 
要 知道 的 ， 我 们 都 会 告诉 你 ， 你 可 以 安心 开始 Clojure 之 旅 。 

除了 从 Lisp 继 承 的 强大 编程 技术 ，Clojure 还 增添 了 一 些 令 人 惊叹 的 前 沿 技 术 。 这 种 组 合 让 
Clojure 从 JVM 语 言 中 脱 括 而 出 ， 成 为 应 用 程序 开发 的 请 人 人 选择。 

Clojure 中 的 并 发 工具 包 和 数据 结构 就 是 一 项 新 技术 ,并 发 抽象 层 让 程序 员 可 以 写 出 更 加 安全 
的 多 线程 代码 。 它 和 Clojure 的 序列 抽象 层 ( 对 集合 和 数据 结构 上 的 不 同 看 法 ) 相 结 合 ， 为 开发 人 
员 提 供 了 非常 强大 的 工具 箱 。 

相 测 握 这 些 力 量 ， 先 要 了 解 Clojure 跟 Java 在 编程 方式 上 截然 不 同 的 理念 。 这 种 差异 使 得 
Clojure 学 起 来 很 有 趣 ， 并 且 很 可 能 会 改变 你 的 思考 方式 。 不 管 你 用 的 是 什么 语言 ， 学 习 Clojure 
都 会 让 你 成 为 更 好 的 程序 员 。 

我 们 一 开始 会 先 讨论 Clojure 处 理 状态 和 变量 的 方式 。 在 给 出 一 些 简 单 的 例子 后 , 会 介绍 这 门 
语言 的 基本 词汇 表 一 一 用 来 构建 语言 其 余部 分 的 特殊 形态 我们 将 深入 到 Clojure 的 语法 中 ， 丁 解 
它 的 数据 结构 、 循 环 和 函数 。 然 后 介绍 序列 , 这 是 Clojure 最 强 的 抽象 概念 之 一 。 我 们 会 用 两 个 非 
常 引 人 注 目的 特性 来 给 这 一 章 收尾 ， 跟 Java 的 紧密 集成 以 及 Clojure 尺 人 的 并 发 支持 。 


10.1 Clojure 介绍 
我 们 先 来 看 Clojure 跟 Java 在 理念 上 最 重要 的 差别 ， 即 对 状态 ， 变 量 和 存储 的 不 同 认识 。 如 图 
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10-1 所 示 , Java( 跟 Groovy 和 Scala 一 样 ) 有 一 个 内 存 和 状态 模型 , 把 变量 当 作 保存 可 变 内 容 的 “使 


oe” ( 内 存 位 置 及 
sOmeInt / 号 


图 10-1 命令 式 语 言 的 内 存 使 用 


而 Clojure 认 为 值 才 是 真正 重要 的 概念 。 值 可 以 是 数字 、 字 符 串 、 回 量 、 映 射 、 集 合 ， 或 其 他 
任何 东西 。 一 旦 创建， 值 就 再 也 不 会 改变 。 这 一 点 直 的 非常 重要 .所 以 我 们 要 再 重复 一 次 。 一 宇 
创建 ，Clojure 的 值 就 不 能 再 变 了 ， 因 为 它们 是 不 可 变 的 。 

这 就 是 说 命令 式 语言 那 种 装着 可 变 内 容 的 盒子 模型 不 是 Clojure 思 考 问 题 的 方式 。 图 10-2 是 
Clojure 处 理 状 态 和 内 存 的 方式 。 它 在 名 字 和 值 之 间 创 建 了 一 个 关联 关系 。 


图 10-2 Clojure 的 内 存 使 用 


这 就 是 绑 定 ， 通 过 特殊 形式 (def) 建立 。Clojure 中 的 特殊 形式 相当 于 Java 的 关键 字 ， 但 请 注 
意 ，Clojure 中 的 术语 “关键 字 ” 含 义 不 同 ， 稍 后 我 们 会 介绍 。 

(def) 的 句法 是 : 

(def< 名 杯 > < 值 >) 

如 果 你 觉得 这 个 句法 看 起 来 有 点 怪异 ,不 要 担心 ， 这 完全 是 Lisp 的 普通 句法 ， 你 很 快 就 会 习 
惯 的 。 现 在 你 可 以 假装 是 在 调用 下 面 这 样 一 个 方法 ， 只 是 括号 的 位 置 不 太一 样 ; 

det(< 上 名 称 >，< 值 >) 


接 下 来 我 们 要 在 Clojure 的 区 互 式 环境 中 写 一 个 久 经 考验 的 例子 ， 演 示 一 下 (def) 的 用 法 。 


10.1.1 Clojure 的 Hello World 
如 果 你 还 没 装 Clojure， 请 参见 附录 D。 然 后 切换 到 Clojure 所 在 的 目录 ， 运 行 如 下 命令 : 


Java -cp clojure.jar clojure.main 

这 个 命令 会 启动 Clojure 的 REPL 环 境 。 在 编写 Clojure 代 码 时 ， 你 会 在 这 个 交互 环境 里 花 上 很 
多 时 间 。 

user=> 是 Clojure 的 会 话 提示 符 ， 你 可 以 把 这 个 会 话 环境 当做 高 级 的 调试 环境 ， 或 者 命令 行 
工具 : 
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User=» (def helloe [En [] "Hello world™)) 
#'user/hello 

user=> (hello) 

"Hello world" 


这 段 代 码 一 开始 先 给 标识 符 hel11o 绑 是 一 个 值 。(def) 就 是 用 来 建立 标识 符 ( Clojure 称 为 和 罕 
号 ) 和 值 之 间 的 绑 定 关系 的 。 底 层 实 现 的 时 候 ， 它 也 会 创建 一 个 对 象 var， 用 来 表示 这 种 绑 定 关 
系 ( 和 符号 的 名 字 ). 

那 这 里 绑 定 的 值 是 什么 ” 这 个 仁 是 ; 

{fn [] "Heéelle world"™) 

这 是 一 个 函数 ， 在 Clojure 中 也 是 一 个 纯正 的 值 ( 因此 也 是 不 可 变 的 ) 这 个 图 数 设 有 参数 ， 
退回 字符 串 "Hello world"。 

绪 定 之 后 ， 可 以 用 (hello) 执 行 。Clojure 运 行 时 会 输出 该 晒 数 的 计算 绪 末 ， 也 就 是 "Hello 
world"s 

现在 ， 应 该 录 人 这 个 例子 ( 如 果 你 还 设 做 )， 看 看 它 的 表现 是 不 是 跟 我 们 说 的 一 样 。 完 成 之 
后 ， 我 们 就 可 以 继续 探索 了 。 


10.1.2 REPL 入 | 


在 REPL 中 可 以 输入 Clojiure 代 码 ， 也 可 以 执行 Clojure 丽 数 。 它 是 个 交互 式 环境 ， 而 且 在 前 面 
得 出 的 计算 结果 不 会 被 丢掉 。 可 以 用 它 做 探索 式 编程 ， 我 们 会 在 10.5.4 节 讨论 这 种 编程 方式 ， 基 
本 就 是 不 断 试验 代 码 。 用 Clojure 开 发 经 常 都 是 先 在 REPL 里 把 代码 调 好 ， 然 后 用 正确 的 构件 措 出 
越 来 越 大 的 国 数 。 

马上 看 一 个 例子 。 先 声明 ， 再 次 调用 aef 可 以 改变 符号 和 值 的 绑 定 关系 , 我 们 在 REPL 中 看 一 
下 。 代 码 中 用 的 实际 上 是 (aef) 的 变 体 (defn): 

user=> (hello) 

"Hello world" 

User=> (defn hello [] "Soodnight Moon") 

#'user/hello 

uBer=s (hello) 

"Goodnight Moon" 


注意 ，helLlo 最 初 的 绑 定 关系 一 下 都 在 ， 直 到 被 你 改 挥 ， 这 是 REPL 的 一 个 关键 特性 。 这 还 
是 状态 ， 只 不 过 换 了 个 说 法 ， 变 成 了 哪个 符号 绑 定 到 哪个 值 上 , 并 且 这 个 状态 存在 于 用 户 输 入 的 
不 同行 间 。 

Clojure 中 没有 可 变 状 态 , 但 有 可 以 改变 绑 定 值 的 符号 。Clojure 不 是 让 “内 存 盒 子 ” 中 的 内 容 
改变 ， 而 是 让 符 导 绑 和 定 到 不 同 的 不 可 变 值 上 。 换 句 话说 就 是 在 程序 的 生命 期 内 ，vazr 可 以 指向 不 
同 的 值 。 请 参见 图 10-3。 


注意 可 变 状态 和 不 同 绑 定 两 者 之 间 的 区 别 很 微妙 ， 但 这 个 概念 很 重要 ,一 定 要 学 握 。 要 记 住 ， 
可 变 状态 是 指 人 金子 中 的 内 容 变 了 ， 而 重新 绑 定 是 指 在 不 同时 间 指 向 不 同 的 金子 。 


时 6 
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一 一 ; 


图 10-3 可 以 改变 的 Clojure 绑 定 


这 段 代码 中 还 激进 了 男 一 个 Clojure 概 念 ，“ 定 义 函数 ” 宏 (defn) 。 宏 是 类 Lisp 语 言 
念 之 一 ， 其 核心 思想 是 内 团结 构 和 普通 代码 之 间 的 区 别 应 该 尽 可 能 小 。 

用 宏 可 以 创建 跟 内 置 语法 类 似 的 形式 。 创 建安 是 高 级 话题 , 但 掌握 了 它 之 后 ， 你 就 能 制造 出 
非常 强大 的 工具 。 

这 就 是 说 语言 真正 的 原 语 系统 ( 特殊 形式 ) 可 以 用 一 种 几乎 无 法 察觉 的 方式 构建 起 整个 语言 
的 核心 。 宏 (defn) 就 是 这 种 构建 的 产物 。 它 只 是 将 函数 值 绑 定 到 符号 的 相对 简单 的 方法 ( 当然 ， 
要 创建 合适 的 var )。 


关键 概 


10.1.3 犯 了 乌 误 
如 果 你 犯错 了 , 会 坚 么 样 ? 比如 你 漏 掉 了 [] ( 函数 声明 的 一 部 分 , 表明 这 个 国 数 设 有 参数 )。 


uUBer=> (ldefn hello "Goodnight Mocnn”") 

#'User/hello 

UB6er=» (hello) 

java.lang.IllegalArgument Exception: Wrong number of args (0) passed to: 

uaers$hello (NO SOURCE FILE:0) 

所 有 后 果 只 是 hel1lo 标 识 符 绑 定 到 了 一 个 未 知 的 东西 上 . 你 可 以 在 REPL 中 重新 绑 定 来 修复 它 : 

uaer=> (defn hello [] (println "Dydh da an NMornl |) 

as ; "Hello World" in Cornish 

#'uUser/hello 

user=> (hello) 

Dydh da an NMor 

nil 

USeEr=> 

跟 你 猜 的 一 样 ， 上 面 这 段 代码 中 的 分 号 ( ; ) 表示 直到 行 尾 的 内 容 都 是 注释 ，(pzintlnl) 是 
输出 字符 串 的 是 数 。 注 意 看 (print1in), 它 跟 所 有 晴 数 一 样 , 返回 了 一 个 值 , 在 函数 执行 结束 后 
回 显 到 REPL 中 。 结 果 值 是 nil1， 相 当 于 Java 里 的 nu11。 


10.1.4 ”学 着 去 爱 括 号 


奇 思 妙 想 和 幽默 感 是 程序 员 文 化 不 可 或 缺 的 一 部 分 。 说 Lisp 是 “很 多 烦人 的 傻 括号 ”的 缩写 
就 是 个 很 古老 的 笑话 。 其 实 Lisp 是 列表 处 理 ( List Processing ) 的 缩写 ， 直 相 就 是 这 么 平淡 无 奇 。 
很 多 Lisp 程 序 员 都 用 这 个 笑话 自嘲 ， 因 为 它 确实 败 到 了 Lisp 语 法 的 痛处 。 


1 于 
OEEEE 
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实际 上 , 这 个 障碍 被 夸大 了 。Lisp 句 法 的 确 特 立 独 行 , 但 也 不 像 看 起 来 那么 碍 手 碍 脚 。 男 外 ， 
Clojure 还 为 减轻 入 门 的 障碍 做 了 几 项 创新 。 

我 们 再 看 一 下 Hello World。 调 用 返回 “Hello World” 的 函数 写成 : 

(hello) 

用 Java 写 应 该 是 这 样 ( 假设 你 已 经 在 某 个 类 里 定义 了 hello 方 法 ): 

hellol); 

但 Clojure 的 表达 式 不 是 myFunction (some0bj) ， 而 是 (myFunction some0bj)。 这 种 写 
法 叫 波兰 表示 法 ， 因 为 它 是 19 世 纪 的 波兰 数学 家 发 明 的 。 

如 果 你 研究 过 编译 原理 ， 可 能 想 知 道 这 是 否 和 抽象 语法 树 ( AST ) 之 类 的 概念 有 关 。 简 单 地 
说 是 “有 ”。 可 以 证 明 , 用 波兰 表示 法 ( Lisp 程 序 员 通常 管 它 叫 s 表 达 式 ) 写成 的 Clojure 或 其 他 Lisp 
程序 是 其 简单 直接 的 AST 表 示 。 

你 可 以 认为 Lisp 程 序 是 直接 用 AST 写 的 。Lisp 程 序 的 数据 结构 表示 和 代码 没有 本 质 上 的 差别 ， 
所 以 代码 和 数据 是 完全 可 以 互 换 的 。 这 也 是 Clojure 的 表示 法 看 起 来 有 点 奇怪 的 原因 一 一 类 Lisp 语 
言 用 它 来 模糊 内 置 的 原生 代码 、 用 户 代码 和 类 库 代 码 之 间 的 区 别 。 对 于 Java 程 序 员 来 说 ， 这 股 强 
大 力量 对 他 们 的 引力 要 远 远 超过 稍微 有 点 古怪 的 语法 。 

让 我 们 更 深入 地 学 一 些 Clojure 语 法 ， 然 后 用 它 写 一 些 真 正 的 程序 。 


10.2 寻找 Clojure: 语法 和 语义 


我 们 上 一 节 介 绍 了 (def) 和 (fn) 两 个 特殊 形式 ( special form )。 这 里 还 有 几 个 需要 你 马上 掌 
握 的 特殊 形式 ， 它 们 构成 了 语言 的 基础 词汇 表 。Clojure 中 还 有 大 量 实用 的 形式 和 宏 ， 用 得 越 多 ， 
认识 会 越 来 越 深 刻 。 

Clojure 中 的 因数 非常 多 ， 托 它们 的 福 ， 你 能 想到 的 任务 很 多 都 可 以 用 Clojure 完 成 。 不 要 因此 
而 诅 袁 ， 你 应 该 感到 庆幸 。 你 要 十 的 活 大 部 分 都 有 人 替 你 二 了 了 ， 不 该 高 兴 吗 ? 


我 们 在 这 一 节 会 讨论 特殊 形式 的 基本 工作 集 ， 人 然后 是 Clojure 的 原生 数 提 
的 集合 ), 之 后 会 接着 讨论 Clojure 代 码 的 自然 编写 风格 一 一 以 函数 而 不 是 变量 为 中 心 。JVM 面 向 
对 象 的 性 质 在 底层 还 会 存在 ， 但 Clojure 强 调 消 数 的 那 种 力量 在 纯粹 的 面向 对 象 语 言 中 表现 得 不 
太 有 明显 。 


10.2.1 特殊 形式 新 手 营 


表 10-1 纵 出 了 一 些 最 芝 用 的 Clojure 特 殊 形式 。 你 现在 最 好 快速 地 把 这 张 表 过 一 遍 , 然 后 在 10.3 
节 遇 到 具体 例子 时 再 回来 看 看 。 

这 个 特殊 形式 列表 不 算 许 尽 ， 并 且 其 中 很 多 特殊 形式 都 有 多 种 用 法 。 表 10-1 中 只 是 它们 的 基 
本 用 例 ， 而 且 都 不 全 面 。 


1 入 
Yr 


类 型 (相当 于 Java 
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表 10-1 Clojure 一 些 基本 的 特殊 形式 
特殊 形式 含 痰 : 

(def< 符 号 > < 值 ?>) 把 符号 乡 到 值 上 (如 果 有 的 话 ) 。 如 有 必要 创建 与 符号 对 应 的 var 

(En< 名 称 >? [< 参数 >*]< 表 达 式 >*) 返回 带 有 特定 参数 的 函数 值 ， 并 把 它们 应 用 到 表达 趟 上 。 通 前 中 (def) 相 缩 台 ， 
变 成 形式 (defn) 

(if<test> <then> <else>?) 旭 果 test 的 计算 结果 为 tue， 计 算 then 并 产 出 其 结果 。 否 则 计算 else 井 产 出 其 
结果 ， 当 然 ， 前 提 是 else 存在 

(1et[< 绑 定 >*] < 表达 式 >*) 给 局 部 名 称 分 配 别 名 值 ， 并 隐 式 定 关 一 个 作用 域 。 使 得 在 let 作 用 域内 的 所 有 表 
达 式 都 能 绪 得 该 别名 

(do< 表 达 式 >*) 按 顺 序 计算 表述 式 的 值 ， 并 产 出 最 后 一 个 的 结果 

(quote< 形 式 >) 站 原样 返回 形式 (不 经 计算 ) 。 它 只 能 接受 一 个 形式 参数 ,其 他 的 参数 全 都 会 被 
忽略 

(var< 特 号 >) 返回 与 符号 对 应 的 var (返回 一 个 Clojure JVM 对 象 ， 不 是 值 ) 


现在 你 对 一 些 特殊 形式 的 基本 语法 有 进一步 的 了 解 了 ， 让 我 们 转 去 看 看 Clojure 的 数据 结构 
吧 ， 也 看 看 它们 怎么 操作 数据 。 


10.2.2 列表、 向 量 、 映 射 和 集 


Clojure 中 有 几 个 原生 数据 类 型 。 用 的 最 多 的 是 列表 list )， 即 单 向 链表 。 

列表 通常 都 用 括号 围 起 来 ， 因 为 形式 一 般 也 是 用 圆 括 号 ， 所 以 这 算是 一 个 轻微 的 语法 障碍 。 
况且 插 写 还 用 来 调用 函数 。 所 以 初学 者 经 常会 犯 下 面 这 种 错误 ; 

1:7 Users=s> (1 2 3) 


java. lang .Classcaat Exception: java.lang.lnteger cannot be cast to 
clojure.lang.IFn (repl-1:7) 


之 所 以 会 出 错 ， 是 因为 Clojure 中 的 值 非常 灵活 , 它 希 望 第 一 个 参数 是 函数 值 ( 或 绑 定 到 函数 
值 上 的 符号 )， 把 2 和 3 当做 这 个 函数 的 参数 。 可 在 上 例 中 1 不 是 函数 值 ， 所 以 Clojure 无 法 编译 。 按 
我 们 的 说 法 ， 这 个 s 表 达 式 是 无 效 的 。 只 有 有 效 的 s 表 达 式 才能 作为 Clojure 形 式 。 

解决 办 法 是 用 (quote) 形 式 ， 它 的 缩写 是 '。 所 以 我 们 可 以 用 两 种 方式 定义 列表 : 


ll:22 USer=» + (1 2 3) 


(二 有 3} 
l:23 Userss (guote (1 2 3)) 
(1 2 3) 


(quote) 以 一 种 特殊 的 方式 处 理 它 的 参数 。 具 体 来 说 就 是 它 不 会 计算 参数 ， 所 以 第 一 个 参数 
不 是 函数 值 也 没 问题 。 

Clojure 的 向 量 ( vector ) 跟 数 组 类 似 ， 实 际 上 ， 基 本 上 可 以 把 Clojure 列 表 等 同 于 Java 的 
LinkedList， 疝 量 等 同 于 arrayList。 向 量 可 以 用 方 括号 表示 ， 所 以 下 面 这 些 定义 都 一 样 : 


l:4 Usera=s (Vector 1 2 3) 


[1 2 3] 

:Ss User=s lvec "1 2 3)) 
[1 2 3] 

1:6 USer=»s [1 2 3] 

[1 2 3] 


1 于 
和 ss 
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在 前 面 声 明 Hello World 和 其 他 函数 时 ， 就 是 用 向 量 来 表示 函数 的 参数 。 注 意 ，(wvec) 形式 以 


一 个 列表 为 参数 ， 并 用 这 个 列表 创建 向 量 ， 而 (vector) 形 式 以 多 个 独立 符号 为 参数 ， 并 返回 包 
含 它 们 的 向 量 。 

函数 (nth) 有 两 个 参数 ， 集合 和 索引 。 它 跟 ]Java 中 List 接 口 的 get () 方 法 类 似 。 可 以 用 在 向 
明和 列表 上 ， 也 可 以 用 在 Java 和 集合 甚 至 字符 串 (字符 的 集合 ) 上 ,请 看 下 例 ; 

li:7 UBer=> (nth ‘(1 2 3) 工 | 

2 


Clojure 也 支持 映射 (map， 相 当 于 Java 的 HaspMap )， 定 义 很 简单 ， 
{keyl valuel Kew3 "value2)} 

从 映射 里 取 值 也 非常 简单 : 

UBer=»> (def foo ("aaa" "111" "bbb" "2222"}) 
#'uUsSer/foo 

USer=s foo 

(naaan "111", "bbb" "2222" |】 

USer=> (foo "aaa") 

“过 是 二 

Clojure 把 前 面 带 冒 号 的 映射 键 称 为 “关键 字 ”: 

l:24 User=>»> ldef martijn { :name "Martijn Verburg", 
= :city "London", :area "Highbury"}) 
#'user/martijn 

l1:25 UsSer=> (ll:name martli]n) 

"Martijn Verburg" 

1:26 USer=»> (martijn :area) 

"Highbury”" 

:2 USEr=y> :Fe 

: area 

1:28 USeEr=> :foo 


:二 GD 

关于 关键 字 ， 请 记 住 下 面 这 些 知 识 点 。 

D Clojure 的 关键 字 是 只 有 一 个 参数 的 函数 ， 其 参数 必须 是 映射 。 

口 在 映射 上 调用 这 个 函数 会 返回 映射 里 与 该 关键 字 函 数 对 应 的 值 。 

口 关键 字 的 使 用 遵循 语法 对 称 性 规则 ， 即 (my-map :key) 和 (:key my-map) 都 是 合法 的 。 

口 关键 字 作 为 值 使 用 时 返回 自身 。 

口 关键 字 在 使 用 之 前 无 需 声明 或 aef。 

DClojiure 中 的 因数 也 是 值 ， 因 此 可 以 放 在 映射 里 当 键 用 。 

口 可 以 用 逗号 (但 设 必 要 ) 来 分 隔 键 / 值 对 ， 因 为 Clojure 会 把 它们 当做 空格 处 理 。 

口 除了 关键 字 ， 其 他 符号 也 能 用 在 映射 里 做 键 ， 但 关键 字 太 好 用 了 ， 所 以 我 们 要 特别 提出 

来 ， 你 应 该 把 它 用 在 自己 的 代码 中 。 

除了 映射 字面 值 ，Clojure 还 有 个 (map) 函数 。 但 不 要 上 当 , 它 不 像 (1ist)，(map) 函数 不 会 
产生 映射 。 而 是 对 集合 中 的 元 素 轮番 应 用 其 参数 中 的 函数 ， 并 用 返回 的 新 值 建 立 一 个 新 集合 ( 实 
际 上 是 Clojure 序 列 ， 请 参见 10.4 节 )。 


1 于 
OEEEE 
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:27 User=s (def ben {:name "Ben Evana", :city "London", :area "Holloway"}) 
user /ben 

:28 User=>s (def authors [ben martijn]j) 

!'uBer/authors 

:29 user=> (map (fn [y] (:name y)} authors) 

"Ben Evans" "Martijn Verburg"!) 


(map) 还 有 别 的 形式 ， 可 以 一 次 处 理 多 个 集合 ,但 一 次 输入 一 个 集合 的 形式 最 常用 。 
Clojure 也 支持 集 ( set )， 跟 Java 的 Hashset 很 像 。 它 的 缩写 形式 是 : 
#{"apple" "pair" "Peach" | 
这 些 数据 结构 是 构建 Clojure 程 序 的 基础 。 

Java 土 闭 可 能 会 感到 吃惊 ， 拓 然 一 直 没 有 提 到 对 象 。 这 不 是 说 Clojure 不 是 面向 对 象 的 ， 但 它 
对 面向 对 象 的 观点 的 确 和 Java 不 一 样 。Java 认 为 世界 是 由 封 著 了 数据 和 代码 的 静态 数据 类 型 组 成 
的 。 而 Clojure 强 调 孙 数 和 形式 ， 尽 管 这 些 在 底层 都 是 由 JVM 上 的 对 象 实现 的 。 

Clojure 和 Java 在 世界 观 上 的 差别 最 终 会 体现 在 代码 里 。 要 充分 理解 Clojure 的 观点 ， 必 须 用 
Clojure 写 些 程序 ， 并 弄 明 白 相 比 Java 的 面向 对 象 结构 它 有 哪些 优势 。 


10.2.3 ”数学 运算 、 相 等 和 其 他 操作 

Clojure 没 有 Java 里 那 种 意义 上 的 操作 符 。 所 以 怎么 才能 ， 比 如 说 ， 让 两 个 数 相 加 呢 ? 在 Java 
里 这 很 容易 : 

3 + 4 

但 Clojure 没 有 操作 符 ， 只 能 用 郴 数 : 

(aaaQ 3 4) 

这 也 挺 好 , 但 我 们 可 以 做 得 更 好 。 因 为 Clojure 里 没有 操作 符 ， 所 以 我 们 不 用 为 它们 保留 任何 
字符 。 这 就 是 说 Clojure 的 函数 名 称 可 以 更 加 稀奇 古怪 ， 所 以 我 们 可 以 这 样 写 ": 


ai 


(+ 3 卫生 | 
Clojure 辆 数 一 般 都 支持 变 参 (和 参数 数量 可 变 )， 比 如 还 可 以 这 样 : 
{+ 1 2 3) 


这 个 运算 结果 是 6。 

Clojure 的 相等 形式 ( 相当 于 Java 里 的 equals () 和 == ) 状况 稍微 有 点 复杂 。Clojure 有 两 个 跟 
相等 相关 的 形式 : (=) 和 (identical?)。 注 意 它们 的 名 字 ， 这 全 都 是 因为 Clojure 不 用 为 操作 符 
保留 字符 。 另 外 ，(=) 也 是 等 号 ， 而 不 是 赋值 符号 。 

下 面 这 段 代 码 设 置 了 一 个 列表 1ist-inc 和 一 个 向 量 vect-int， 并 比较 它们 是 香 相 等 : 


OD 例子 中 的 (+) 是 clojure. core 命 名 空间 下 的 函数 ， 能够 接受 0 到 任意 数目 的 参数 ， 假 如 没有 参数 ， 则 返回 90。 所 以 
虽然 Clojure 没 有 操作 符 ， 但 有 很 多 提供 了 操作 符 功 能 的 核心 函数 ， 所 以 体 大 可 不 必 担 心 怎么 计算 3 * 4， 用 早已 
准备 好 的 冰 数 (* 3 4) 就 行 了 。 一 一 译 者 注 
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1:1 user=»> (def list-int + (1 2 3 4)) 

#'UuUSerF/list-=-int 

1:2 User=> (def vect-int (vec list-1int)| 

#'UuUSer/vect-int 

1:3 Uger=s {= Vect-int ligst-1int) 

ee (identical? vect-int ligst-1int) 

{false 

(=) 形式 会 检查 集合 是 否 由 相同 的 对 象 以 相同 的 顺序 组 成 的 (1ist-int 和 vect-int 符 合 这 
一 要 求 )， 而 (identical?) 会 检查 它们 是 否 真 的 是 同一 个 对 象 。 

你 可 能 也 注意 到 了 ， 符 号 名 称 都 没有 用 驼峰 式 大 小 写 。 这 在 Clojure 中 很 常见 ， 符 号 通常 都 
用 小 写 ， UT 


i 和 页 上 ” i a | | 全 二 上 直下 LE 人 有 ny 
= ~ 由 i . 下 
和 出 Ee 4 | ph | Bi | fe = 二: | 由 与 首 | 
pv Nn 人 1 Er lt, + Ed te 下 TO ey i df Tr. 
.| 本 rr- a | sm 上 和 中 Ps 时 | #1 上 -二 - ef 
] @ | Fr | 站 


站 二 | h es , bt ， = = en 
LC fe 二 四 一 rs Ta 和) ep 


tt 三 个 值 表 a A 很 多 动态 语言 才 这 梓 
但 对 于 Java 程 序 员 来 说 这 有 ， 点 奇怪 。 


掌握 了 基本 的 数据 结构 和 撞 作 符 ， 让 我 们 把 之 前 见 过 的 特殊 形式 和 函数 拼 到 一 起 ， 写 一 个 稍 
徽 长 点 的 Clojure 上 图 数 吧 。 
10.3 ”使 用 函数 和 循环 


从 本 节 开 始 , 我 们 会 接触 到 Clojure 中 一 些 实质 性 的 内 容 。 从 编写 函数 处 理 数据 开始 ， 让 你 看 
到 Clojure 对 函数 的 重视 程度 。 接 着 介绍 循环 结构 ， 以 及 读 取 项 (reader ) 宏和 派发 (dispatch ) 形 
式 。 最 后 ,我们 会 以 Clojure 的 函数 式 编程 和 闭 包 作为 本 节 的 收尾 。 

举例 说 明 是 好 办 法 , 所 以 我 们 先 来 几 个 简单 的 例子 , 然后 朝 Clojure 提 供 的 强大 函数 式 编 程 技 
术 进 发 。 


10.3.1 一 些 简单 的 Clojure 函数 
代码 清单 10-1 中 定义 了 三 个 函数 。 其 中 两 个 是 非常 简单 的 单 参 函数 ， 另 一 个 稍 徽 有 点 复杂 。 
代码 清单 10-1 定义 简单 的 函数 


(defn const-funl [Y] 21) 


Ley , 本 
和 中 pp EE pF 

i | 1 | n 四 全 全 Pm 

和 六 于 开 芋 a rT 


(defn ident=-fun [y] Y) 
Idefn liast-maker-fun [x £] 
(‘map (fn [2] {let [Ww 2] 

(list w (f w)) 
}} x}} 


中 驼峰 式 大 小 写 ( Camel-Case ) 一 词 来 自 Perl 语 言 中 普遍 使 用 的 大 小 写 混 全 格式， 而 Lary Wall 等 人 所 著 的 畅销 书 


Programming Perl: Unmatched power for text processing and scripting ( O'Reilly,， 2012 ) 的 封面 图 片 正 基 一 匹 骄 避 。 
一 一 译 者 注 
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在 这 段 代 码 中 ， (const-funl) 接 受 一 个 参数 ， 返回 1， (ident-fun) 接 受 一 个 数值 并 返回 
数值 本 身 , 数学 家 会 管 它们 叫 常量 函数 和 量 等 函数 。 还 有 , 函数 定义 中 使 用 向 量 表示 函数 的 参数 ， 
(1et) 形式 中 用 的 也 是 了 向量 。 

第 三 个 阻 数 比 较 复杂 , 函数 (1ist-maker-fun) 有 了 两 个 参数 : 
量 x， 第 二 个 一 定 是 果 数 。 

我 们 来 看 一 下 如 何 使 用 1ist-maker-fun， 如 代码 清单 10-2 所 示 。 


代码 清单 10-2 ”使 用 函数 

Users=s> (liast-maker-fun ["a"] conet=- 上 tunl) 

【区 时 西昌 1)) 

UsSers> (list-maker-fun ["a”" "b"] const-funl) 

("a" 1}) ("bb" 1)) 

USBer=»> (list-maker-fun [2 1 3] ident-fun) 

(C2 2) (1 1}) (3 3 

UBer=> (list-maker-fun [2 1 3) "a") 

Java.lang.ClassCast Exception: Java.lang.Sstring cannot be cast to 

clojure .lang. IFn 

把 这 些 表达 式 逆 到 REPL 中 实际 上 是 和 Clojure 的 编 详 带 交 互 。 表 达 式 (list-maker-fun [2 
1 3] "a") 之 所 以 无 法 编译 ， 是 因为 (1ist-maker-fun) 的 第 二 个 参数 应 该 是 函数 ， 而 字符 串 
显然 不 是 。 看 到 10.5 节 你 就 会 知道 ， 对 于 VM 来 说 ，Clojure 轴 数 是 实现 J 了 clojure.1lang .IFn 的 
对 象 。 

这 个 例子 表明 在 跟 REPL 交 互 时 仍然 会 涉及 一 些 静态 类 型 问题 . 因为 Clojure 不 是 解释 型 语言 。 
即便 是 在 REPL 中 ， 输 入 的 每 个 Clojure 形 式 都 会 被 编译 成 JVM 字 节 码 并 连接 到 运行 时 系统 上 。 
Clojure 阴 数 在 定义 完 后 就 被 编译 成 JVM 字 节 码 了 ， 所 以 在 出 现 静态 类 型 冲突 时 VM 会 报 出 
ClassCastException 异 常 。 

代码 清单 10-3 中 的 Clojure 代 但 更 长 。Schwartzian 转 换 可 有 年 头 了 ， 从 20 世 纪 90 年 代 在 Perl 中 
出 现 后 就 一 直 在 用 。 其 基本 思想 是 基于 向 量 中 元 素 的 某 些 属性 对 元 素 进 行 排序 。 排 序 所 依据 的 属 
性 值 是 通过 在 元 素 上 pi 


第 一 个 是 包含 所 处 理 的 值 的 问 


] 定义 n 转 换 。 的 键 控 图 获 是 key-En。 在 真正 调用 (schwartz) 
明 数 时 需要 提供 一 个 用 作 键 控 的 函数 . 代码 清单 10.3 中 用 的 是 我 们 的 老 朋 友 (iaent -fun) 。 


代码 清单 10-3 ”Schwartzian 转 换 
1:65 USer=> (defn achwartz [x key-fnl | 第 三 三 步 
Imap (fn [YY) (nth ¥ 0)) 加 


(Bort-by (tn [t] (nth t 1)) 第 二 步 
Imap (fn [z] (let [w z] 四 i 和 
(list w (key-fn w))) ] 第 一 步 
】 KR} )1}) 


Q "user/schwartz 

1:66 USer=s (gchwartz [2 3 1 5 a) ident= fun) 

( 工 了 3 卫 妆 5 

1:67 USBers=> (apply schwartz [[2 3 1 5 4] taenc-tun]) 
(1 县 3 在 5] 
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口 创建 一 个 包含 键 值 对 的 列表 :; 

口 基于 键 控 肾 数 的 值 对 键 值 对 排序 ; 

口 仅 从 排 好 序 的 键 值 对 列表 中 取出 原始 值 ， 构 建新 列表 ( 并 抛弃 键 控 函数 值 )。 
如 图 10-4 所 示 。 


图 10-4  Schwartzian 转 换 


代码 清单 10-3 中 引信 了 一 个 新 形式 : (sort-by) 。 这 个 函数 有 两 个 参数 ; 一 个 是 用 来 排序 的 
前 数 ， 一 个 是 要 排序 的 向 量 。 还 有 (apply) 形 式 , 它 也 有 两 个 参数 : 一 个 是 要 调用 的 函数 ,一 个 
是 传 给 它 的 向 量 参数 。 

Randall Schwartz 最 初 用 Perl 编 写 Schwartzian 转 换 ( 该 转换 以 他 的 和 名字 命 名 ) 时 在 刻意 模仿 
Lisp。 我 们 现在 又 用 Clojure 编 写 ， 算 是 绕 了 一 圈 又 回来 了 。 挺 有意思， 

Schwartzian 转 换 的 示例 很 实用 , 我 们 稍 后 还 会 用 到 它 。 因 为 它 的 复杂 性 足以 用 来 曾 明 好 几 个 
概念 。 

接 下 来 我 们 来 讨论 下 Clojure 的 循环 ， 可 能 和 你 所 习惯 的 循环 有 点 不 太一 样 。 


10.3.2 ”Clojure 中 的 循环 


Java 里 的 循环 相当 简单 直接 ， 可 选 的 循环 有 for 、while， 还 有 其 他 几 种 。 其 核心 思想 通常 
是 重复 一 组 指令 ， 和 下 到 满足 某 一 条 件 (一 般 用 一 个 可 变 变 量 表 示 )。 

这 对 Clojure 是 个 小 难题 ; 举 个 例子 , 对 于 没有 可 变 变 量 作为 循环 案 引 的 Clojure, 怎么 表示 for 
循环 呢 ? 在 传统 的 Lisp 中 通常 用 递归 形式 实现 循环 。 但 JVM 不 能 保证 尾 递归 优化 ( Scheme 和 其 他 
Lisp 语 言 有 这 种 要 求 )， 所 以 在 Clojure 中 用 递归 可 能 会 导致 栈 溢出 ，。 

而 Clojure 有 不 会 增加 栈 空间 占用 的 结构 。 最 常用 的 是 loop-recur， 下 面 的 代码 展示 了 如 何 
用 loop-recur 构 建 一 个 和 for 循 环 类 似 的 结构 ，。 
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(defn like-=-for [counter)] 
[loop [ctr counter] 
(println ctr) 
(4 {< ctr 10) 
(recur (ine ctr)})) 
ctr 
))) 

(loop) 形式 以 包 会 符号 局 部 名 称 的 问 量 为 参数 一 一 像 (1et) 定 义 的 别名 。 然 后 当 执 行 到 
(recur) 形 式 时 (本 例 中 只 有 ctr 别 名 小 于 10 才 会 执行 该 形式 )， 它 会 将 控制 分 文 退 回 到 (locp) 
形式 中 , 但 指定 了 新 的 值 。 这 样 我 们 就 可 以 搭建 循环 式 结构 ( 比如 for 和 while 循 环 ), 但 实现 中 
仍 有 递归 的 味道 。 

现在 我 们 转 入 下 一 主题 ， 看 一 看 Clojure 语 法 的 简写 ， 帮 你 把 程序 写 得 更 短 、 更 精炼 。 


10.3.3” 读 取 器 宏和 派发 器 

Clojure 有 些 让 很 多 Java 程 序 员 吃惊 的 语法 特性 。 其 中 之 一 是 没有 操作 符 。 它 的 副作用 是 放宽 
了 Java 对 能 用 在 名 称 中 的 字符 的 限制 。 你 已 经 见 过 像 (identical?) 这 样 的 函数 了 ， 这 在 Java 中 
是 非法 的 ， 但 对 于 哪些 字符 不 能 用 在 符号 中 ， 我 们 还 没有 说 明 。 

表 10-2 列 出 了 不 能 用 在 Clojure 符 号 中 的 字符 。Clojure 分 析 和 保留 了 这 些 字符 日 用 , 它们 通常 
被 称 为 读 取 器 宏 。 


表 10-2 读 取 器 宏 
字 符 名 称 / 含义 
| 与 展开 为 (guote) ， 产 出 不 进行 计算 的 形式 
; 注释 标记 直到 行 尾 的 注释 ， 就 像 Java 里 的 / 
\ 子 符 产生 一 个 字面 字符 
8 解 引用 ”展开 为 (deref)， 接 受 var 对 象 并 返回 对 象 中 的 值 ( 跟 (var) 形 式 的 操作 相反 ) 。 在 事务 
内 存 上 下 文中 还 有 其 他 会 史 ( 见 10.6% ) 
无数 据 特 一 个 元 数据 的 映射 附加 到 对 人 象 上 。 请 查阅 Clojure 文 档 了 解 详情 
| 辜 法 引用 经 常用 在 宏 定 义 中 的 引号 形式 ， 不 太 适 合 初 学 者 。 请 查阅 Clojure 文 档 了 解 详情 
派发 有 几 种 不 同 的 子 形 式 ， 见 表 10-3 
根据 # 后 面 的 字符 ， 派 发 读 取 右 宏 有 几 种 不 同 的 于 形式 ， 请 见 表 10-3。 
表 10-3 派发 读 取 器 宏 的 子 形式 
派发 形式 含 义 
展开 为 (var) 
人 创建 一 个 集 字 面值 ， 在 10.2.2 节 中 用 过 
#1() 创建 匿名 丙 数 字面 值 ， 用 在 那些 使 用 (tn) 太 哪 睦 的 地 方 
# 跳 过 下 一 个 形式 。 可 以 用 # ( ... 多 行 ...) 来 创建 多 行 注释 
"三 机 寺 >" 创建 一 个 正则 表达 式 (作为 java.util.regex.Pattern 对 象 ) 
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关于 派发 形式 ， 还 有 几 点 要 提 一 下 。 变 量 引 用 形式 #' 解 释 了 REPL 执 行 (daef) 之 后 的 表现 : 
1l:49 user=> (def someSymbol) 
#4'user/someSymbol 


(def) 形式 返回 新 创建 的 var 对 象 , 命名 为 someSymbol, 驻 留 在 当前 的 命名 空间 中 ( 就 是 用 
户 所 在 的 REPL )， 所 上 #' user/someSymbol 是 (def) 壕 回 的 完整 值 。 

匿名 国 数 字面 值 也 是 减少 繁琐 代码 的 创新 。 它 省 略 了 和 参数 向 量 , 用 一 种 特殊 的 语法 让 Clojure 
读 取 器 推断 函数 宇 面值 需要 多 少 个 参数 。 

代码 清单 10-4 是 我 们 用 这 个 语法 重 写 的 Schwartzian 转 换 。 


代码 清单 10-4 重 写 Schwartzian 转 换 
Idefn schwartz [x £] 


Iimap 站 (nth $1 0) i | 
(gort -by #(nth %1 1) | 讶 名 函数 字面 值 


(imap #(let [Ww %®1] 
(list w (f w)) 
X11}) 


用 $1L 当 做 冰 数 字面 值 参数 的 占 位 符 ( 后 续 参 数 可 以 用 $2 、$3 等 ) 真 的 很 好 ， 这 样 的 代码 也 
更 容易 看 恒 。 这 种 显而易见 的 线索 对 程 订 员 很 有 帮助 , 就 像 你 在 9.3.6 节 见 过 的 Scala 函 数字 面值 里 
的 箭头 得 号 一 样 。 

Clojure 疗 重 依 赖 于 以 函数 为 基本 计算 单元 的 概念 ， 而 不 像 Java 以 对 象 为 语言 的 根本 。 这 种 方 
式 目 然 会 寻 回 图 数 式 编程 ， 也 就 是 我 们 的 下 一 主题 。 


10.3.4 国 数 式 编程 和 闭 包 


我 们 现在 要 进 人 开 怖 的 Clojure 函 数 式 编程 世界 。 或 者 ， 我 们 没有 ， 因 为 它 不 忍 怖 。 实 际 上 ， 
我 们 这 一 整 章 都 在 学 习 晴 数 式 编程 ， 只 是 没 告诉 你 ， 怕 把 你 吓 跑 。 

7.3.2 节 中 说 过 ， 盯 数 式 编程 意味 着 函数 是 一 个 值 。 果 数 可 以 传递 ， 放 在 变量 中 操作 ， 就 像 2 
或 "hello" 一 样 。 但 那 又 怎么 样 ? 我 们 回头 看 看 第 一 个 例子 ; (def hello (fn [] "Hello 
worla")) 。 我 们 创建 了 一 个 函数 ( 没有 参数 ， 返 回 字 符 串 "Hello worlda* )， 把 它 绪 定 到 符号 
hel1o 上 。 国 数 仅仅 是 个 值 ， 本 质 上 跟 2 这 种 值 设 什 么 区 别 。 

在 10.3.1 节 ， 我 们 以 Schwartzian 转 换 为 例 介绍 了 以 另外 一 个 函数 为 输入 值 的 函数 。 这 也 不 过 
是 一 个 以 特定 类 型 为 输入 参数 的 函数 ， 唯 一 的 区 别 不 过 是 这 个 类 型 是 函数 。 

关于 闭 包 呢 ? 它们 真 的 很 嫩 怖 ， 是 不 是 ?” 哦 ， 还 好 吧 。 我 们 来 看 一 个 简单 的 例子 ， 这 应 该 能 
让 你 想起 我 们 做 过 的 一 些 Scala 例 子 ， 

1:5 User=> ldefn adder [constToAdd] #(+* constToAhdd $%1)) 

HH 'uger/adder 

1:6 USer=> [def plus2 (adder 2)) 

#'usSer/plus2 
1:7 user=»> [plus2 3) 

; 
了 


:8 USer=> 1:9 SETe=> (plus2 5) 


1 入 
三 
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上 例 中 先 定义 了 (adadaer) 函数 。 这 是 一 个 构造 其 他 函数 的 函数 。 如 果 你 熟悉 Java 硬 言 的 工厂 
方法 模式 ， 可 以 把 它 当 成 Clojure 的 工厂 方法 实现 。 以 其 他 函数 为 函数 的 返回 值 没 什么 好 奇怪 的 ， 
这 是 将 函数 作为 普通 值 这 一 概念 的 重要 体现 。 

这 个 例子 给 匿名 函数 用 了 编写 的 #() 形 式 。 函 数 (adder) 接受 一 个 数值 参数 并 返回 一 个 函 
数 ， 并 且 返 回 的 是 带 一 个 参数 的 因数 。 

然后 用 (aaaer) 定 义 了 一 个 新 形式 : (plus2)。 这 个 函数 接受 一 个 参数 ,并 在 这 个 参数 上 加 
2。 这 就 是 说 绑 定 到 (aadaer) 内 部 的 constToadd 的 值 是 2。 现 在 我 们 来 构造 一 个 新 明 数 ， 


:13 UsSer=> (def plus3 ladder 3)) 


1 
#'uUser/plus3 

1:14 user=> (Plus3 4) 
了 

| 

6 


:15 User=> (plus2 4) 


这 段 代 码 表 明 你 还 可 以 再 构造 其 他 函数 (plus3) ， 绑 定 不 同 的 值 到 constToadaa 上 。 我 们 说 
困 数 (plus3) 和 (plus2) 已 经 从 它们 所 在 的 环境 中 捕获 或 “封装 ”了 一 个 值 "。 需 要 注意 的 是 
(plus3) 和 (plus2) 捕 获 的 值 是 不 同 的 ， 并 且 定 义 (plus3) 对 (plus2) 捕 获 的 值 没有 影响 

在 自身 环境 内 “封装 ”一 些 值 的 函数 称 为 闭 包 ，(plus2) 和 (plus3) 就 是 闭 包 , 在 支持 闭 包 
的 语言 中 ， 用 一 个 制造 者 函数 构造 并 返回 男 一 个 封装 了 一 些 东 西 的 函数 非常 普遍 。 

接 下 来 我 们 要 讨论 Clojure 中 一 个 强大 的 特性 : 序列 。 它 们 使 用 了 跟 Java 的 集合 或 迭代 器 类 似 
的 东西 ， 但 有 些 不 同 的 属性 。 在 代码 中 使 用 序列 最 能 体现 Clojure 语 言 的 力量 ， 对 于 习惯 了 Java 处 
理 方式 的 程序 员 ，Clojure 的 处 理 方式 会 让 你 耳目 一 新 。 


10.4 ”Clojure 序列 


看 下 而 这 段 代码 中 的 Java 迭 代 器 。 这 是 使 用 迭代 器 的 老 套 路 了 了。 实际 上 ，Java 5 里 的 for 循 环 
在 底层 也 会 被 转换 成 这 种 实现 ， 

Collection<String> CGC = | 

for (Iterator<String> it = c.iterator(); it.hasNext ();}) | 

String str = it.next(); 

| 

对 于 简单 集合 的 循环 处 理 这 就 够 了 ， 比 如 set 或 List。 但 Iterator 接 口 只 有 next() 和 
hasNext () 方 法 ， 加 上 一 个 可 选 的 zemove () 方 法 。 

1. 残缺 的 Java 和 迭代 器 

然而 Java 友 代 逢 还 有 缺陷 。 和 迭代 肯 接 口 所 提供 的 集合 交互 方法 满足 不 了 需求 。 用 Iterator 
只 能 做 两 件 事 : 

口 查看 集合 中 是 否 还 有 更 多 的 元 么 ; 


QD 此 处 的 环境 即 指 函 数 (aader)， 而 捕获 的 值 即 绑 定 到 conscTroada 的 值 。 一 一 译 者 注 
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口 取出 下 一 个 元 素 ， 并 把 迭代 器 向 前 推进 。 

Iterator 最 主要 的 问题 是 把 取得 下 一 个 元 素 和 向 前 推进 合 在 了 一 起 ( 如 图 10-5 所 示 ) 这 意 
味 着 无 法 先 对 集合 中 的 下 一 个 元 素 进行 检查 , 然后 再 决定 它 是 需要 特殊 处 理 , 还 是 完好 无 损 地 取 
出 去 。 


图 10-5 Java 从 代 器 的 性 质 


从 玩 代 器 中 取出 下 一 元 素 的 行为 改变 了 它 的 状态 。 也 就 是 说 可 变 已 经 内 建 在 Java 处 理 集合 和 
夺 代 器 的 方法 中 了 ， 因 此 不 可 能 用 它 构建 出 强健 的 多 路 解决 方案 。 
2. Clojure 的 键 抽象 
Clojure 采 用 了 不 同 的 方式 。Clojure 与 Java 中 的 集合 与 进 代 器 相对 应 的 核心 概念 是 序列 
( sequence )， 或 者 简称 seq。 它 基本 上 是 把 两 个 Java 类 的 一 些 特性 集成 到 了 一 个 概念 里 。 这 样 做 的 
动机 有 三 个 : 
口 更 强 健 的 只 代 髓 ， 特 别 是 对 于 多 路 算法 而 言 ; 
口 不 可 变 能 力 ， 可 以 安全 地 在 函数 间 传 递 厅 列 ; 
口 实现 懒 序列 的 可 能 性 ( 后 面 还 会 详细 讨论 )。 
表 10-4 中 列 出 了 跟 序 列 相关 的 一 些 核心 功能 。 这 些 函 数 都 不 会 改变 它们 的 参数 ， 如 果 它 们 需 
要 返回 不 同 的 值 ， 那 会 是 一 个 不 同 的 序列 。 
表 10-4 ”基本 的 序列 函数 
函数 作 用 
(seq <coll>) 返回 一 个 序列 ， 作 为 所 操作 集合 的 “视图 ” 
(first <coll>) 返回 集合 的 第 一 个 元 素 , 如 有 必要 , 先 在 其 上 调用 (eeg) 。 如 果 集 侣 为 nil, 则 返回 nil 
(Fest <coll>) 返回 从 集合 中 去 掉 第 一 个 元 素 后 得 到 的 新 序列 。 如 果 集 合 为 nil1， 则 返回 nil 
lseq? <o>) 如 果 o 是 一 个 序列 则 返回 true ( 也 就 是 实现 了 了 ISeqg ) 
(cons <elt> <coll>) 在 集合 前 面 增加 新 元 素 ， 并 返回 由 此 得 到 的 序列 
(conj <coll> <elt>)  。 返回 将 新 元 素 加 到 合适 一 端 ( 向 基 的 尾 端 和 列表 的 头 ) 的 新 集合 
levery? <pred-fn> 如 果 (prea- fn) 对 集合 中 的 每 个 元 素 都 返回 于 辑 真 ， 则 返回 true 


COll>s) 


这 里 有 几 个 例子 


1 于 
OEEEE 


260 第 10 章 Clojure: 更 安全 地 编程 


1:1 user=» (reast '(1 2 3})) 


{2 3) 

li:2 USer=s (first ‘(1 2 3}) 
1 

1:3 user=»> (Test [1 2 3]) 
(2 3) 

1:13 UsSer=> (Segq ()) 

riil 

1:14 user=»> (seq []) 

nil 

1:15 USer=> (cong 1 [2 3)) 
(1 卫 了 3) 

1:16 User=> (levery? is-prime [2 3 5 7 11]) 
true 


有 一 点 要 重点 关注 一 下 ,列表 是 目 身 的 序列 ， 而 向 量 不 是 。 因 此 从 理论 上 来 说 ， 不 能 在 向 量 上 
调用 (zest) 。 可 实际 上 是 可 以 的 ， 因 为 (rest) 在 操作 回 量 之 前 先 在 其 上 调用 了 (seq) 。 这 是 序列 
结构 中 普遍 存在 的 属性 : 很 多 序列 函数 都 会 接受 比 序列 更 通用 的 对 象 , 并 在 开始 之 前 先 调 用 (seq) 。 

我 们 在 这 一 节 中 准备 探索 seq 的 一 些 基 本 属性 和 用 法 ， 尤 其 会 重点 关注 司 序 列 和 变 参 函数 。 
其 中 第 一 个 概念 “ 懒 "， 是 Java 中 不 太 会 涉及 的 编程 技术 ”， 所 以 对 你 来 说 它 可 能 比较 新 颖 。 现 在 
我 们 就 来 看 一 下 吧 。 


10.4.1 懒 序列 


在 编程 请 言 里 ， 情 是 一 个 强大 的 概念 。 其 基本 思想 是 将 表达 式 的 计算 推迟 到 需要 时 。 体 现在 
Clojure 中 就 是 序列 可 以 不 是 完整 的 值 列表 ,其 中 的 值 可 以 在 被 请 求 时 取得 ( 比如 根据 需要 通过 调 
用 肾 数 生成 它们 )。 

在 Java 中 ， 要 满足 这 样 的 想法 就 得 菲 定制 的 List 实 现 , 而 且 要 写 大 量 的 套路 化 代码 才 可 能 实 
现 。 用 Clojure 中 的 宏 呈 要 做 一 点 儿 工 作 就 能 创建 出 懒 序 列 。 

想 一 想 怎 么 才能 创建 出 一 个 懒惰 的 、 可 能 包含 无 限 数量 值 的 序列 。 很 明显 ， 用 函数 来 生成 序 
列 内 的 元 素 。 这 个 函数 应 该 做 两 件 事 : 

口 返回 序列 中 的 下 一 个 元 于 ; 

口 接受 数量 固定 、 有 限 的 参数 。 

数学 家 会 说 这 样 一 个 靖 数 定义 的 是 递归 关系 ,并 且 这 样 的 关系 用 道 归 的 方式 处 理 再 恰当 不 
过 了 。 

假设 有 一 台 在 栈 空 间 和 其 他 能 力 上 都 不 受 限 制 的 机 器 , 并 且 可 以 执行 两 个 线程 : 一 个 用 来 生 
成 无 限 的 序列 ， 男 外 一 个 使 用 该 序列 。 那 我 们 就 可 以 在 生成 线程 里 用 递归 定义 懒 序列 ， 类似 下 面 
这 段 伪 代码 : 


(deftn infinite-seg <Vvec-args> 
{let [new-val (seqg-fn <vec-args>)] 
(cons new-val (inftinite-segqg new-Vec-args>))}) 


由 用 过 Hibernate 的 人 一 定 知道 傅 加 载 ( 因为 它 原 来 经 常 爆 异 常 )， 其 基本 思路 “延迟 ”四 懒 是 一 样 的 。 一 一 译 者 注 
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实际 上 在 Clojure 中 这 是 行 不 通 的 , 因为 (infinite-seq) 上 的 递归 会 导致 栈 溢出 。 但 要 是 加 
上 一 个 结构 ， 告 诉 Clojure 不 要 疯狂 递归 ， 仅 根据 需要 进行 处 理 ， 是 可 以 做 到 的 。 

不 仅 如 此 ， 你 还 能 在 一 个 线程 内 做 到 这 一 点 ， 如 下 例 所 示 。 代 码 清单 10-5 中 为 某 个 数 k 和 定义 
了 懒 序列 k，k+1, k+2， ...。 


代码 清单 10-5 ” 懒 序列 的 例子 
(defn next-big-n [In (let [new-val (+ 1 n})] lazv-aad 标 记 
(lazy- 3eq Rs 
(cons new-val (next -big-n new-val)) : 
))) 无 限 递归 


(defn natural-k [Kk] 
(concat [k] (next -big-=-n k))})) 
Pe : ] concat 限 制 递归 

1:57 Ugera=s (take 10 (natural=-k 3)) 

(二 5S 10 11 1z) 

(lazy-seq) 此 式 是 关键 , 它 标 记 了 发生 无 限 递 归 的 点 , 还 有 (concat)， 可 以 安全 地 处 理 递 
归 。 然后 你 就 可 以 用 (take) 形 式 从 懒 友 列 中 取出 所 需 的 元 素 了 , 这 个 基本 上 是 用 (next-big-n) 
形式 定义 的 。 

懒 序列 是 极其 强大 的 特性 ， 实 践 会 告诉 你 它们 是 Clojure 军 火 库 中 的 强大 武器 。 


10.4.2 ”序列 和 变 参 函数 


Clojure 困 数 有 一 个 强大 的 特性 ， 它 天 生 就 具备 参数 数量 可 变 的 能 力 ， 有 时 称 为 函数 的 变 元 
( arity )。 参数 数量 可 变 的 力 数 称 为 变 参 函数 (variadic )。 

代码 清单 10-1 中 讨论 过 的 函数 (econst-funl) 可 以 作为 一 个 简单 的 例子 。 这 个 函数 接受 一 个 
参数 并 抛弃 它 ， 总 是 返回 值 1。 请 看 传人 多 个 参数 给 (const-fun1) 时 会 发 生 什 和 


1:32 USer=»> (const-funl 2 3| 
java.lang.IllegalAhrgumentException: Wrong number of args (2) passed to: 
Usgersconast-funl (repl-1:32) 


Clojure 编 详 带 仍然 会 对 传 给 (const-fun1) 的 参数 数量 ( 和 类 型 ) 做 一 些 检查 。 对 于 简单 地 
抛弃 所 有 参数 并 返回 一 个 常量 值 的 函数 来 说 , 这 似乎 过 于 严格 了 。 在 Clojure 中 能 接受 任意 数量 参 
数 的 函数 看 起 来 会 是 什么 样 的 呢 ? 

代码 清单 10-6 展 示 本 如 何 实 现 一 个 这 样 的 (const-fun1) 常量 孙 数 。 我 们 管 它 吊 (const- 
fun-arityl), 变 元 的 const-funl。 这 是 在 Clojure 标 准 晒 数 库 中 (constantly) 因数 的 目 产 版 。 


代码 清单 10-6 ”之 有 变 元 的 函数 
l:28 USBer=> (defn const-fun-arityl 
([] 32) 
{ [x]】 1) 带 不 同 签名 的 多 个 daefn 
([x 各 meore] 1) 
) 
#'user/const-fun-arity]l 
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1:33 user=> (lconst-fun-arityl) 


J] 

1:34 USer=> (const-fun-arity] 2) 

1 

1:35 usger=s lconat-fun-arityl 2 3 4) 
1 


这 个 图 数 的 定义 不 是 一 个 参数 同 量 后 跟着 盟 数 行为 的 定义 。 而 是 有 一 系列 这 种 组 合 ， 每 个 组 
合 里 都 是 一 个 参数 向 量 ( 构成 了 这 一 版 本 函数 的 有 效 签 名 ) 和 这 一 版 本 了 晴 数 的 实现 。 

这 跟 Java 的 方法 重 载 类 似 。 传 统 做 法 一 般 是 定义 几 个 特 跌 情况 下 的 形式 (没有 参数、 一 个 或 
两 个 参数 ) 和 最 后 一 个 参数 为 序列 的 额外 形式 。 代 码 清 单 10-6 中 就 是 参数 向 量 为 [x & more] 的 
那个 。& 符 号 表明 这 是 该 函数 的 变 参 版 本 。 

序列 是 Clojure 的 创新 。 实 际 上 ， 用 Clojure 编 程 主要 就 是 要 思考 怎么 用 序列 解决 特定 问题 。 

Clojure 的 男 一 项 重要 创新 是 Clojure 和 Java 的 集成 ， 也 就 是 我 们 下 一 节 的 主题 。 


10.5 ”Clojure 与 Java 的 互 操作 


Clojure 从 一 开始 就 设计 为 JVM 语 言 , 并 且 不 会 对 程序 员 完 全 隐藏 JVM 特 性 。 这些 特 殊 的 设计 
在 几 个 地 方 都 有 体现 。 比 如 在 类 型 系统 层面 ，Clojure 的 列表 和 向 量 都 实现 了 Java 集 人 台 类 库 中 的 标 
准 接 口 List。 男 外 ，Clojure 使 用 Java 的 类 库 非常 容易 ， 反 之 亦 然 。 

这 意味 者 Clojure 程 序 员 可 以 使 用 Java 中 丰富 的 类 库 和 工具 ， 以 及 JVM 的 性 能 和 其 他 特性 。 这 
一 帮会 涉及 这 种 互 操作 性 的 几 方 面 内 容 ， 特 别 是 : 

口 从 Clojure 中 调用 Java; 

口 Java 如 何 见 到 Clojure 函 数 的 类 型 ; 

口 Clojure 代 理 ， 

口 从 Java 中 调用 Clojure。 

我 们 先 看 看 从 Clojure 中 如 何 访 问 Java 方 法 ， 开 始 它 们 的 集成 探索 之 旅 吧 。 


10.5.1 从 Clojure 中 调用 Java 
一 下 这 上 段 在 REPL 中 进行 计算 的 Clojure 代 码 : 


看 
1:16 user=> (defn lenSstr [y] (.length {(.toString vy))})) 
#'user/lenstr 

1:17 uaer=> (schwartz ["bab" "aa® "dgfwag" "droopy"] laenstr) 
("aa" "bab" "dgfwgq" "droopy") 

1:18 Uger=> 


这 段 代码 用 Sechwartzian 转 换 对 一 个 字符 串 向 量 排序 , 排序 标准 是 字符 串 的 长 度 。 其 中 用 到 了 
形式 ( .tostring) 和 (.length) ， 这 都 是 Java 方 法 ， 它 们 是 在 Clojure 对 象 上 调用 的 。 符 号 开始 
部 分 的 句号 .表示 运行 时 应 该 在 下 一 个 参数 上 调用 该 名 称 的 方法 ， 底 层 是 用 ( . ) 宏 实 现 的 。 

所 有 用 (aef) 或 它 的 变 体 定义 的 Clojure 值 都 被 放 在 clojure .1lang .var 实 例 中 , 它 可 以 承载 


只 自在 读书 @3 
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任何 java.lang.object， 所 以 任何 可 以 在 java.lang.0bject 调 用 的 方法 都 可 以 在 Clojure 值 
上 调用 。 另 外 一 些 跟 Java 交 互 的 形式 是 用 来 调用 静态 方法 的 
(System/getProperty "java.vm.version") 


( 此 处 是 调用 system. getProperty!l) ) 和 用 于 访问 静态 公 * 共 变量 ( 比如 销量 ) 的 


Boolean/, TRUE 
在 后 面 两 个 例子 中 已 经 用 到 了 Clojure 命 名 空间 的 概念 . 跟 ]Java 包 的 概念 类 似 , 并 且 常 用 的 Java 
包 都 有 对 应 的 映射 缩写 形式 ， 比 如 前 面 那 些 。 


i gp 闫 际 二 十 汰 辣 二 站 于 汪汪 JVM 不 PPYTTaprer ( 特别 是 
Scheme ) 通 常 做 的 那样 优化 掉 尾 递归 ，。 JVM 上 一 些 其 他 的 Lisp 方 言 觉得 它们 需要 真正 的 尾 递归 ， 
因此 不 准备 把 Lisp 函 数 调用 跟 JVM 方 法 调用 完全 等 同 起 来 。 而 Clo ire 完 全 以 JVM 为 平台 ， 甚至 
不 惜 违背 通常 的 Lisp 实 践 。 


如 果 你 想 创建 一 个 新 的 Java 对 象 实例 并 在 Clojure 中 操作 它 ， 用 (new) 形式 就 可 以 轻松 做 到 。 
它 还 有 个 备 选 的 缩写 形式 ， 在 类 名 之 后 跟 一 个 句号 ， 可 以 归结 为 ( . ) 宏 的 另 一 个 用 法 : 

(import ' (java.util .concurrent CountDownLatch LinkedBlcockingoueue) ) 

(def cdl (new CountDownLatch 2)) 

(def lbqg (LinkedBlockingQueue.})) 


这 里 还 用 了 (import) 形 式 ， 只 用 一 行 就 可 以 导入 一 个 包 的 很 多 Java 类 。 
我 们 在 前 面 提 过 ，Clojure 的 类 型 系统 有 些 地 方 跟 Java 是 一 致 的 ， 我 们 来 看 看 其 中 的 细节 。 


10.5.2 ”Clojure 值 的 Java 类 型 
从 REPL 中 很 容易 看 到 某 些 Clojure 值 的 Java 类 型 : 


1:8 user=»> (.getClase "foo") 
java.lang.String 


1:9 UBer=> (.getClass 2.3) 

java.1lang.Double 

1:10 user=s= (.getClass [1 2 3]) 

clojure. lang.PerslgtentVector 

1:11 user=> (getClass (上 (1 2 3)) 
cloiure, lang. PersistentList 

1:12 User=s (.getCclass (fn [] "Helle world!")) 
usgerS$evalll0s$fn 111 


首先 要 看 到 所 有 Clojure 值 都 是 对 象 , JVM 的 原始 类 型 默认 情况 下 是 不 对 外 的 ( 人 尽 旨 从 性 能 角 
度 来 看 有 办 法 得 到 原始 类 型 )。 如 你 所 料 ， 字 符 串 和 数字 值 直 接 映射 到 对 应 的 Java 引 用 类 型 上 去 
7 (java.lang.String、java.lang .Double 等 ), 


匿名 的 "Hello worldi "函数 的 名 字 表明 它 是 一 个 动态 生成 类 的 实例 。 这 个 类 会 实现 
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clojure.lang.IFn 接 口 ，Clojure 用 该 接口 表明 这 个 值 是 个 函数 ， 你 可 以 把 它 当 做 
java ,util.concurrent 里 的 Callable 接 口 。 

序列 会 实现 clojure .1ang .ISeq 接 口 。 它 们 通常 是 抽象 类 ASeq 或 懒 实现 Lazyseq 的 具体 
于 类 。 

我 们 已 经 看 过 几 种 值 的 类 型 了 ， 但 这 些 值 是 怎么 保存 的 呢 ? 就 像 我 们 在 本 章 一 开始 提 到 的 ， 
(defE) 把 香 与 绑 到 一 个 值 上 ， 这 样 会 创建 一 个 var。 这 些 var 是 clojure ,lang,Var 类 型 ( 它 所 
实现 的 接口 中 也 有 IFn ) 的 对 象 。 


10.5.3 使 用 Clojure 代理 

Clojure 有 一 个 强大 的 (proxy), 你 可 以 用 它 创 建 扩展 Java 类 ( 或 实现 接口 ) 的 Clojure 对 象 。 
比如 代码 清单 10-7 重 新 实现 了 之 前 的 一 个 例子 ( 代码 清单 4-13 ), 由 于 Clojure 语 法 更 加 紧凑 ,所 以 
这 个 例子 的 核心 代码 只 有 一 点 。 
代码 清单 10-7 重 温 调度 执行 者 


(import ‘(java.util,concurrent Executors LinkedBlockingQueue TimeUnit)) 


(def atpe (Executors/newSscheduledThreadPool 2)) 
(def lbq (LinkedBlockinagQueue.})) gTBPE 工 厂 方 法 
z | 1] , | 


lI(def msgRdr (proxy [Runnablel < 
(run [] {.toSstring (.poll lbq}))}) | 定义 匿名 的 Runmable 实 现 
)) | | 


(def rdrHnd]l 
“ (scheduleAtFixedRate stpe msgRAar 10 10 Timetmnit /MILLISECONDS})) 


(proxy) 的 一 般 形 式 是 : 

(proxy [< 超 类 /接口 >] [<args>] < 命名 函数 的 实现 >+) 

第 一 个 向 量 参数 是 这 个 代理 类 应 该 实现 的 接口 。 如 果 这 个 代理 还 要 扩展 Java 类 【如 果 可 以 的 
话 ， 当 然 ， 只 能 扩展 一 个 Java 类 )， 这 个 类 名 必须 是 向 量 中 的 第 一 个 元 素 。 

第 二 个 癌 量 参数 包含 传 给 超 类 构造 方法 的 参数 。 这 个 向 量 经 常 是 空 的 ， 并 且 如 果 (proxy) 
形式 只 是 实现 Java 接 口 的 话 ， 那 它 肯 定 是 空 的 。 

这 两 个 参数 之 后 是 一 个 或 多 个 表示 单个 方法 实现 的 形式 ， 按 接口 的 要 求 或 超 类 指定 的 
实现 。 

用 (proxy) 形式 可 以 做 出 任何 Java 接 口 的 简单 实现 。 这 促成 了 一 种 吸引 人 的 可 能 性 ， 用 
Clojure REPL 作 为 实验 Java 和 JVM 人 代码 的 扩展 游戏 床 。 


10.5.4 用 REPL 做 探索 式 编程 


探索 式 编程 的 核心 思想 是 减少 要 编写 的 代码 最 ， 因 为 Clojure 的 语法 和 REPL 提 供 的 实时 互动 
环境 ，REPL 不 仅 是 探索 Clojure 编 程 的 理想 环境 ， 也 是 学 习 Java 类 库 的 极 佳 选 择 。 
我 们 来 看 一 下 Java 列 表 实 现 。 它 们 都 有 返回 Iterator 类 型 对 象 的 iterator () 方 法 。 但 
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Tterator 是 个 接口 ， 所 以 你 可 能 对 真正 的 实现 类 型 感到 好 奇 。 用 REPL 很 容易 找 出 答案 : 
1:41] User=> (import {java.util ArrayList LinkedList)) 
java.util .LinkedList 
1:42 USer=» (.getClass (.iterator (ArrayLiast.))) 
java.util.ArrayLiatsItr 
1:43 USer=> (.getClass (.iterator (LinkedList.)})}) 
Java.util .LinkedListsListItr 


(import) 形 式 从 java .util 包 中 导 人 J 两 个 类 ,然后 在 REPL 内 用 Java 的 getclass() 方 法 。 
可 以 看 到 进 代 此 实 际 上 是 内 部 类 提供 的 。 也 许 你 不 应 该 对 此 感到 吃惊 , 因为 我 们 在 10.4 广 讨论 过 ， 
夺 代 项 和 它 们 的 集合 绑 定 很 紧密 ， 所 以 它们 也 许 需要 了 解 这 些 集 合 的 内 部 实现 细节 。 

在 前 面 这 个 例子 中 值得 注意 的 是 ,我 们 一 个 Clojure 结 构 也 没 用 ， 只 用 了 一 点 语法 。 我 们 操作 
的 所 有 东西 实际 上 都 是 Java 结 构 。 尽 管 如 此 ， 我 们 还 是 假设 你 想 用 不 同 的 方式 ， 在 Java 程 序 里 用 
Clojure。 下 一 节 将 会 问 你 展示 如 何 实现 这 一 目的 。 


10.5.5 在 Java 中 使 用 Clojure 


Clojure 的 类 型 系统 跟 Java 高 度 一 致 。Clojure 数 据 结构 全 是 真正 的 Java 集 合 ， 都 实现 了 对 应 接 
口 的 所 有 必需 部 分 ,因为 接口 的 可 选 部 分 一 般 都 跟 修改 数据 结构 有 关 , 而 Clojure 数 据 结 构 不 可 变 ， 
所 以 一 般 者 设 实现 。 
类 型 系统 的 一 致 性 使 得 在 Java 程 序 里 使 用 Clojure 数 据 结构 成 为 可 能 。Clojure 自 身 的 性 质 加 强 
了 这 种 可 行 性 一 一 它 是 采用 调用 机 制 的 JVM 编 译 型 语言 。 这 最 大 限度 地 减少 了 运行 时 的 问题 , 意 
味 着 从 Clojure 中 得 到 的 类 几乎 跟 其 他 任何 Java 类 一 样 。 解 释 型 语言 跟 Java 的 互 操作 会 更 加 困难 ， 
并且 通常 需要 最 基本 的 非 Java 碚 言 运 行 时 支持 。 
下 面 这 个 例子 展示 了 Clojure 的 seq 结 构 如 何 用 在 一 个 普通 的 Java 字 符 串 中 。 要 运行 这 段 代 码 ， 
需要 把 clojure.jar 放 在 classpath 上: 
ISedq aed = StringSeq.create("foobar"); 
while {seqg != null) 1 
Object first = seq.firest{(}.; 
System.out .println(t"Seaq: "+ Beg +'" } firast: "+ first); 
SEeg = Seg.next |); 
上 面 的 代码 使 用 了 stringsea 类 中 的 工厂 方法 create()。 它 给 出 了 字符 串 中 字符 序列 的 seq 
视图 。first() 和 next |() 方 法 返回 新 值 ， 而 不 是 修改 已 有 的 seq， 就 跟 我 们 在 10.4 节 讨论 的 一 样 。 
截止 目前 我 们 只 是 在 处 理 单线 程 的 Clojure 代 码 。 下 一 节 我 们 要 谈论 Clojure 中 的 并 发 。 特 别 是 
Clojure 对 状态 和 可 变性 的 处 理 方式 ， 这 使 得 它 的 并 发 模型 跟 Java 的 差别 很 大 。 
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Java 的 状态 模型 从 根本 上 来 说 是 基于 对 象 可 变 思 想 的 。 正 如 第 4 章 中 所 提 及 的 ， 这 会 直接 导 
致 并 发 代码 的 安全 问题 。 在 某 一 线程 修改 对 象 的 状态 时 ， 为 了 防止 其 他 线程 看 到 对 象 的 中 间 ( 即 
不 一 致 ) 状态 ,需要 引信 相当 复杂 的 锁 策 略 。 这 些 策略 理解 难 ， 调 试 难 ， 测 试 更 难 。 
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Clojure 的 并 发 概念 在 某 些 方面 不 像 Java 中 那么 底层 。 比 如 说 ， 由 Clojure 运 行 时 管理 线程 池 的 
使 用 ( 开发 人 员 在 这 方面 几乎 或 根本 不 能 控制 ) 看 起 来 可 能 有 点 奇怪 ,但 是 让 平台 ( 此 处 即 Clojure 
运行 时 ) 细致 地 做 好 内 务工 作 的 好 处 在 于 ， 开 发 人 员 可 以 专注 于 更 重要 的 任务 ， 比 如 总 体 设计 。 

Clojure 的 指导 思想 是 默认 把 线程 彼此 陋 开 , 这 种 实现 并 发 安全 的 办 法 由 来 已 久 。 假 定 “ 没 有 
共享 资源 ”的 基线 和 采用 不 可 变 值 使 Clojure 避 开 了 很 多 Java 所 面临 的 问题 ， 从 而 可 以 专注 于 为 并 
发 编程 安全 地 共享 状态 的 方法 。 


注意 ”为 了 帮助 提升 安全 性 ，Clojure 的 运行 时 提供 了 线程 协调 机 制 ， 我 们 强烈 建议 你 使 用 这 些 
机 制 ， 而 不 是 用 Java 的 避 例 或 构造 自己 的 并 发 结构 。 

实际 上 ，cClojure 用 不 同 的 方法 实现 了 不 同 的 并 发 模型 : 未 来 式 (fture )、 并 行 调用 (pcall )、 

引用 形式 ( ref ) 和 代理 (agent )。 且 听 我 们 一 一 道 来 ， 先 从 最 简单 的 开始 。 


10.6.1 未 来 式 与 井 行 调用 


第 一 个 也 是 最 明显 的 一 个 状态 分 享 办 法 就 是 不 分 享 。 实 际 上 , 我 们 一 直 使 用 的 Clojure 结 构 var 
本 质 上 是 不 可 以 共享 的 。 如 果 两 个 不 同 的 线程 继 了 乔 了 和 名字 相同 的 var， 并 在 线程 里 重新 绑 定 了 它 ， 
那 绑 定 只 在 这 些 线程 内 部 可 见 ， 绝 不 可 能 被 其 他 线程 共享 。 

可 以 利用 Clojure 跟 Java 的 紧密 结合 启动 新 线程 ， 也 就 是 说 在 Clojure 中 写 Java 并 发 代码 非常 容 
易 。 但 其 中 有 些 抽象 在 Clojure 中 有 更 干净 的 形式 。 比 如 对 于 第 4 章 介绍 的 Java 未 来 式 ( Future )， 
Clojure 提 供 了 非常 干净 的 方式 。 代 码 清单 10-8 是 个 简单 的 例子 。 


代码 清单 10-8 ”Clojure 中 的 Future 
User=> (def simple-future 
(future (ds 

(println "Line 0") 

(Thread/sleep 10000) 

(println "Line 1") 

(Thread/sleep 10000) 

(println "Line 2")))) 
#'user/simple-future | 马上 开始 执行 
Line 0 可 加 
USer=> {future-done? simple-future) 

USsSer=> false 


Line 1 

USer=> @gimple-future 

Line 2 | 和 解 引 用 导致 阻塞 
nil 

USeLr=> 


这 段 代码 用 (future) 建立 了 一 个 Future。 创建 之 后 它 马上 就 开始 在 后 台 线 程 中 运行 ,所 以 
在 Clojure REPL 中 看 到 了 输出 Line 0 (然后 是 Line 1 ) 一 一 代码 已 经 开始 在 男 一 个 线程 上 运行 
了 。 接 者 可 以 用 (future-done?) 来 检查 代码 是 理 已 经 运行 完 ， 这 个 调用 是 非 阻塞 的 。 然 而 对 
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future 的 解 引 用 会 阻塞 调用 线程 ， 直 到 函数 完成 。 

这 实际 上 是 Clojure 对 Java Future 的 一 个 瘦 封 装 ， 语 法 更 干净 。Clojure 还 提供 了 对 并 发 程序 
员 非 常 有 帮助 的 辅助 形式 。 有 个 简单 的 函数 是 (pcalls) ， 可 以 接受 数量 可 变 的 零 参 果 数 ， 让 人 它 
们 并 发 执行 。 它 们 在 运行 时 管理 的 线程 池上 执行 ,并 返回 一 个 懒 序 列 结 果 , 试图 访问 序列 中 的 任 
何 还 没完 成 的 元 素 会 导致 访问 线程 被 阻塞 。 

代码 清单 10-9 建 立 了 一 个 单 参 因数 (wait-with-fEor)。 它 用 了 一 个 类 似 10.3.2 节 介绍 过 的 
locop 形 式 。 可 以 用 它 创建 一 些 零 参 困 数 (wait-1L)、(wait-2) 等 ， 并 把 它们 传 给 (pecalls) 。 


代码 清单 10-9 Clojure 中 的 并 行 调用 
USer=> (defn wait-with-for [limit] 
(let [eounter 1] 
(loop [ctr counter)] 

(Thread/sleep S500) 

(println (str "Ctra" ctr)) 

(二 下 (< ctr limit) 

(recur (inc ctr}) 
etr)}})) 

Huser/wait-with-for 
usBer=y (defn wait-=1 [] (wait-with-for 1})) 
USer=s #'user/wait-] 
USer=> (defn wait-=-2 [] (wait-with-for 2}) 
USBET=> 半 'USer/wait-2 
User=> (defn wait-3 [] (wait-with-for 3)) 
USeEr=»> #"'USer/wait-3 
UBer=> (def wait-seg (pcalls wait-]1 wait-2 wait-3)) 
user/wait—3eg 
Ctra=l 
Ctr=l 
| 
Ctrse 
Ctrsz 
在于 呈 导 


USeEr=> (first wait-=seq) 

1 

Ugéer=> (first (next wait-seq)) 
2 


因为 线程 睡眠 值 只 有 500 上 毫秒 ， 等 竺 函数 很 快 就 能 完成 。 通 过 调整 超时 ( 比如 延迟 到 10 秒 )， 
很 容易 验证 由 (pcalls) 返 回 的 懒 序列 wait-segq 是 否 有 上 面 描述 的 那 种 阻塞 行为 。 
对 于 不 需要 共享 状态 的 情况 ,这 种 简单 的 多 线程 结构 挺 好 ,但 在 很 多 应 用 中 ,不 同 的 处 理 线 
程 都 要 在 运行 过 程 中 相互 通信 。Clojure 有 几 个 模型 可 以 处 理 这 种 情况 , 接 下 来 我 们 先 看 看 其 中 的 
-个 : 借助 (ref) 形 式 实现 的 状态 共享 。 


10.6.2 ref 形式 
ref 是 Clojure 在 线程 间 共 享 状 态 的 办 法 。 它们 基于 运行 时 提供 的 一 个 模型 , 在 这 个 模型 中 , 状 
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态 的 改变 要 能 被 多 个 线程 见 到 。 该 模型 在 符号 和 值 之 间 引 人 了 一 个 额外 的 中 间 层 。 也 就 是 说 , 符 
号 绑 定 到 值 的 引用 上 ， 而 不 是 直接 绑 到 值 上 。 这 个 系统 基本 上 是 事务 化 的 ,并 且 由 Clojure 运 行 时 
进行 协调 。 如 图 10-6 所 示 。 

线程 A 线程 B 


50MeVar Otherwar 


bb 
a 


| clojure.lang. Ret ' 


图 10-6 软件 事务 内 存 


这 一 中 间 层 意味 着 改变 或 更 新 ref 之 前 必须 把 它 放 在 一 个 事务 中 。 当 事务 完成 的 时 候 , 或 者 全 
变 了 了 ， 或 者 什么 也 设 变 。 这 跟 数 据 库 中 的 事务 是 类 似 的 。 

这 可 能 有 点 抽象 了 ,所 以 我 们 来 看 一 个 模拟 ATM 的 例子 。 在 Java 中 ， 要 对 所 有 敏感 数据 加 锁 
保护 。 代 码 清单 10-10 是 一 个 简单 的 自动 提 款 机 模型 ， 包 括 锁 ， 


代码 清单 10-10 ”Java 中 的 ATM 模 型 
public class Account | 
private double balance = 0) 
private final String name; 
private final Lock lock = new ReentrantLock!().; 


public Account (String name , double initialBal ){ 
name = name } 
balance = initialBal ; 

} 

public synchronized double getBalance()| 
return balance; 


public synchronized void debit (double debitAmt ) | 
balance -= debitAmt ; 


public String getName() | 
return name:; 


} 


Publie String toString() 1 
return "Account [balance=" + balance + ", name=" + name + "]":; 


} 
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public Lock getLock() { 
return lock; 


| 


} 

public class Debitter implements Runnable { 
private final Account acc:} 
private final CountDownLatch cdl,) 


public Debitter (Account account , CountDownLatch cdl ) |{ 
acc = account ;} 
cdl] = cdl }; 
} 
public void run() | 
double bal = acc.getBalancel).; 
Lock lk = acc.getLock(); 


while (bal > 0) 1 


try 1{ 
Thread.sleep(1); 让 2 
| catch (InterruptedException €) { } 能 在 acc 上 同步 


lk.lock(}); es 

bal = acc.getBalance(), 

if (bal > 0) 1 : pe ! 
acc .debit (1) ; 必须 重新 取得 余额 
bal==; 


| 


1K .unlock!():; 


cdl .countDown'():; 


上 
Account myAcc = new Account ("Test Account", 500 * NUM THRERADS) ; 
CountDownLatch stopl = new CountDownLatch (NUM THREADS), 
for (int i=0; i<NUM THREADS; i++) | 
new Thread (new Debitter (myAcc, stopl)) .start(); 


} 
atopl .await (); 
System.out .println (myAcc); 


再 来 看 看 用 Clojure 怎 么 写 。 先 来 个 单线 程 版 本 。 然 后 我 们 再 开发 一 个 并 发 版 本 跟 单 线程 版 本 
比较 ， 这 样 并 发 代码 应 该 更 容易 理解 。 
代码 清单 10-11 是 单线 程 版 本 。 


有 . i 到 二 下 而 
ee : - i me = - = i Ey | : - 
ss a 1 i | = = 一 | = 和 7 | 
区 T 8 a 上 a 国 | | | 9 
E J = 二 L 1 = 二 " EE EE dS 和 _ 
各 er Tn 人 es el | ry rn eT a i a , | ] < 如 奸 和 站 到 
Re 府 y ’ 


人 


(defn make-new-acc [account-name opening-balance] 
{:name account-name :bal opening-balance}) 


(defn loop-and-debit [account)] 
(loop [acc account] 
(let [balance (:bal acc) my-name (:name acce)] 


由 自在 读书 他 
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(Thread/sleep 1) a 
(if (> balance 0) 用 循环 /递归 代 敬 Java 
(recur [make-new-acc my-name (dec balance)}) 中 的 while 
aACC 
)))) 


(loop-and-debit (make-new-acc "Ben” 50001 1 

这 段 代码 跟 Java 版 比 起 来 非常 紧凑 。 必 须 承 认 ， 这 是 单线 程 的 ， 但 还 是 比 Java 的 代码 少 了 很 
多 。 运 行 代码 会 得 到 期 望 的 结果 ;一 个 余额 为 0 的 acc 上 映射 。 现 在 我 们 看 看 并 发 形式 。 

要 让 这 段 代 码 并 行 ， 需 要 引入 ref。 它 们 是 用 (ref) 形式 创建 的 ， 并 且 类 型 为 clojure. 
lang .Ref 的 JVM 对 象 。 通常 建立 时 会 带 一 个 保存 状态 的 映射 , 此 外 还 需要 (dosync) 形 式 来 设置 
事务 。 在 事务 之 内 , 还 要 用 到 (alter) 形 式 来 修改 ref, 使 用 ref 的 多 线程 ATM 函 数 如 代码 清单 10-12 
所 示 。 


[defn make-new-ace [accoumt -name opening-balancel] 
(ref {:name account-name :bal opening-balance}})) 


(detn alter-ace [acc new-name new-balancel] 


agasoc ace :bal new-balance :name new-name)) 和 
四 必须 返回 值 ， 而 不 是 引用 
(defn loop-and-debit [account] 


{loop [acc account)] 
(let [balance (:bal vacc) 
my-name (:name @acc)] 
(Thread/gleep 1) 
(if 【> balance 0) 
{recur ldosync lalter acc alter-acc my-name ldec balance)) acc)) 
CE 
站 
(def my-acc (make-new-acc "Ben'" S5000)) 
(defn my-=-loop [] ‘(let [the-acc my-acc] 
(loop-and-debit the-acc) 
)) 


(pcalls my-loop my-loop my-loop my-loop my- loop) 

就 像 注释 中 说 的 ， 对 值 进行 操作 的 (alter-acc) 函数 必须 返回 一 个 值 。 所 操作 的 值 是 对 当 
前 事务 中 线程 可 见 的 本 地 值 , 这 称 为 事务 内 的 值 。 返 回 的 值 是 在 变更 函数 返回 之 后 的 ref 新 值 。 在 
退出 (dosync) 所 定义 的 事务 块 之 前 ， 这 个 值 对 外 界 是 不 可 见 的 。 

与 此 同时 ， 其 他 事务 可 能 像 这 个 一 样 也 在 进行 。 如 果 是 这 样 ，Clojure STM 系 统 会 进行 跟踪 ， 
并 且 只 允许 那些 自 开 始 以 来 已 经 提交 过 的 事务 组 成 的 事务 提交 。 如 果 不 一 致 , 它 会 回 演 , 并 且 可 
能 在 得 到 更 新 过 的 状态 后 再 次 党 试 。 

如 果 事 务 做 了 任何 会 产生 副作用 的 事情 ( 比如 日 志文 件 或 其 他 输出 ), 这 个 重 试行 为 可 能 
会 引发 问题 。 让 事务 化 部 分 在 函数 式 编程 中 ( 即 没 有 副作用 ) 尽 可 能 地 保持 简单 纯粹 是 你 的 
责任 。 


1 入 
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对 于 某 些 多 线程 方式 而 言 , 这 种 持 乐 观 态度 的 事务 行为 看 起 来 可 能 是 相当 重量 级 的 做 法 。 有 
些 并 发 应 用 只 需 偶 尔 在 线程 间 进 行 通信 ， 并且 是 以 相当 不 对 称 的 风格 。 症 运 的 是 ，Clojure 提 人 供 了 
另外 一 种 更 好 地 体现 “过 后 就 愁 “ 原 则 的 并 发 机 制 ， 这 也 是 我 们 下 一 了 的 主题 。 


10.6.3 代理 


代理 是 Clojure 中 异步 的 、 面 向 消息 的 并 发 原 语 。Clojure 代 理 不 是 共享 状态 ,而 是 属于 男 外 一 
个 线程 的 一 点 儿 状 态 ， 但 它 会 从 男 外 一 个 线程 中 接收 消息 (以 函数 的 形式 )，。 这 乍 看 起 来 可 能 是 
个 奇怪 的 想法 ， 尽 管 过 到 过 Scala 的 actor 之 后 这 种 感觉 可 能 会 少 一 点 。 


“我 离 它们 太 远 了 ， 只 能 把 礼物 装 进 包 昌 和 寄 给 它们 ,” 她 想 ,“ 这 也 术 滑 征 了 ， 给 自 
己 的 双 脚 送礼 物 还 需要 邮寄 ! 地 址 写 起 来 就 更 有 趣 了 1” 
一 一 《爱丽 丝 梦 游 仙 境 》， 刘 易 斯 . 卡 罗 尔 


应 用 到 代理 上 的 函数 在 代理 的 线程 上 运行 。 这 个 线程 是 由 Clojure 运 行 时 管理 的 , 在 一 个 程序 
员 通 常 无 法 访问 的 线程 池 里 。 运 行 时 还 会 保证 代理 中 那些 可 以 被 外 界 看 到 的 值 是 孤立 的 和 原子 
的 。 这 就 是 说 用 户 代 码 只 会 见 到 状态 修改 之 前 或 之 后 的 代理 值 。 

代码 清单 10-13 是 个 简单 的 代理 例子 ， 跟 用 来 讨论 future 的 例子 类 似 。 


代码 清单 10-13 ”Clojure 代 理 
(defn wait-and-log [coll str-to-add] 
(do (Thread/sleep 10000) 
(let [my=coll (conj coll str-to-add)] 
(Thread/sleep 10000) 
(conij] my-coll gtr-to-add)}})) 


(def str-coll (agent [] ) ) 
(send str-coll wait-and-log "foo") 
Batr-coll 
send 调 用 派发 了 一 个 (wait-and-1og) 调 用 给 代理 , 通过 使 用 REPL 解 引用 , 结果 就 像 承 诺 的 
那样 ， 你 绝 不 会 看 到 代理 的 中 间 状 态 一 一 只 有 最 后 的 状态 出 现 了 (字符 串 "foo' 被 添加 了 两 次 )。 
实际 上 ， 代 码 清单 10-13 上 的 (send) 调 用 很 容易 让 人 联想 到 爱丽 丝 的 脚 的 地 址 。 刘 易 斯 . 卡 
罗 和 尔 很 可 能 是 用 Clojure 代 码 写 的 地 址 ， 
党 丽 毕 的 右 牌 收 
壁炉 前 的 毛重 上 
靠近 持 板 
【( 带 去 爱丽 丝 的 爱 ) 
在 你 认为 一 个 人 的 脚 是 身体 的 有 机 组 成 时 ， 这 的 确 挺 怪异 的 。 同 样 ， 发 消息 给 Clojure 管 理 的 
线程 池 中 一 个 线程 上 的 代理 看 起 来 也 挺 怪 异 的 两 个 线程 还 共享 一 个 地 址 空间 。 但 你 目前 多 次 明 
到 的 一 个 并 发 主题 就 是 如 果 它 能 让 用 法 更 加 简单 清晰 ， 额 外 的 复杂 性 可 能 是 件 好 事 。 
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10.7 小结 


作为 一 门 语言 ，Clojure 可 以 说 是 我 们 见 过 的 几 门 语言 中 跟 Java 差 别 最 大 的 。 它 对 Lisp 的 传承 、 
对 不 可 变性 的 强调 以 及 独特 的 编程 方式 , 让 它 看 起 来 变 成 了 完全 独立 的 语言 。 但 它 和 JVM 的 紧密 
结合 、 与 类 型 系统 的 一 致 性 ( 即便 它 提供 了 序列 等 替代 方案 )， 还 有 探索 式 编程 的 能 力 ， 让 它 成 
为 与 Java 互 补 性 非常 强 的 一 门 语言 。 

任何 地 方 的 协同 都 没有 Clojure 运 行 时 对 线程 和 并 发 底层 特性 的 代理 控制 更 清晰 。 这 让 程序 员 
可 以 放手 去 关注 多 线程 的 设计 和 高 层 问 题 。 这 就 跟 Java 的 垃圾 收集 设施 可 以 让 你 无 需 关 心 内 存 管 
理 的 细节 一 样 。 

本 部 分 研究 的 不 同 语 言 间 的 差别 展示 了 Java 平 台 的 进化 能 力 , 并 且 证 明了 它 仍 然 是 应 用 开发 
的 理想 目标 。 这 也 是 对 JVM 灵 活性 和 性 能 的 证 明 。 

在 本 书 的 最 后 一 部 分 , 我 们 会 向 你 展示 三 门 新 语言 为 软件 工程 实践 提供 的 新 方式 。 下 一 章 全 
部 是 关于 测试 驱动 开发 的 内 容 一 一 你 在 Java 世 界 中 很 可 能 已 经 碰 到 过 这 一 主题 了 。 但 Groovy、 
Scala 和 Clojure 提 供 了 全 新 的 视角 ， 有 望 丽 固 和 加 强 你 已 经 知 谊 的 那些 东西 。 
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第 四 部 分 


多 语种 项 目 开发 


在 最 后 一 部 分 ， 我 们 会 把 已 经 学 到 的 平台 和 多 语言 编程 知识 应 用 到 现代 软件 开发 中 最 常见 
和 最 重要 的 技术 上 ， 

要 成 为 一 名 优秀 的 Java 开发 人 员 ， 不 仅仅 是 掌 提 JVM 和 它 上 面 跑 的 语言 那么 简单 。 要 成 
功 欧 付 软件 ， 还 要 遵循 业界 最 佳 实践 。 幸 好 ， 这 些 实践 中 有 相当 一 部 分 是 从 Java 生态 系统 中 开 
始 的 ， 所 以 我 们 有 很 多 东西 可 以 聊 。 

我 们 会 用 一 整 章 的 内 容 讨 论 训 试 驱动 开发 (TDD) 的 基础 知识 ， 以 及 如 何 把 测试 概念 应 用 
到 极其 复 厅 的 测试 场景 中 。 男 一 章 会 集中 讨论 如 何 将 正规 的 构建 生命 周期 引信 构建 流程 中 ， 包 
括 持 续集 成 技术 。 这 两 章 会 介绍 一 些 工 具 ， 比 如 用 于 测试 的 JUnit、 用 于 构建 的 Maven， 以 及 用 
于 持续 集成 的 Jenkins。 

我 们 还 会 讨论 Java 7 时代 的 Web 开发 ， 会 涉及 为 项 目 选 择 最 适合 框架 的 标准 ， 还 有 如 何在 
这 个 环境 中 快速 开发 。 

如 末 你 看 过 第 三 部 分 ， 应 该 了 解 非 Java 语言 在 TDD、 构 建生 命 周期 和 快速 Web 开发 领 
域 都 有 举足轻重 的 作用 。 无 论 是 用 于 TDD 的 ScalaTest 框架 ， 或 者 用 于 构建 Web 应 用 的 Grails 
(Groovyy) 和 Compojure (Clojure)〉 框架 ，Java/JVM 生态 系统 中 的 很 多 方面 都 受到 了 这 些 新 语言 
的 影响 。 

我 们 会 向 你 展示 如 何 把 新 语言 的 力量 作用 到 你 所 熟悉 的 软件 开发 工艺 上 。 与 JVM 坚实 的 基 
础 和 Java 生态 系统 结合 为 一 个 整体 ， 你 会 发 现 那些 接受 多 语言 观点 的 开发 人 员 可 能 会 收获 颇 丰 . 

节 后 一 草 我 们 会 看 一 看 平台 的 未 来 ， 并 预测 一 下 将 来 。 第 四 部 分 全 是 前 沿 内 容 ， 所 以 现在 
就 让 我 们 翻 开 新 的 一 页 ， 向 着 地 平 线 推进 吧 ! 
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测试 驱动 开 如 ， 


本 章 内 容 
口 A 


ise 虚设 、 伪 装 、 存 根 和 模拟 


口 用 内 存 数据 库 测 试 DAO 人 代码 
口 用 Mockito 覆 拟 子 系统 


测试 驱动 开发 (TDD ) 进入 软件 开发 行业 已 经 有 相当 长 的 时 间 了 。 它 的 基本 前 提 是 在 编写 真 
正 的 功能 实现 代码 之 前 先 写 测试 代码 ， 然 后 根据 需要 重 构 实现 代码 。 比 如 要 写 一 段 拼接 两 个 
string 对 和 象 ( "foo" 和 "bar" ) 的 实现 代码 , 应 该 先 瑟 测试 代码 (测试 结果 必须 等 于 "foobar" )， 
以 确保 你 能 判断 实现 是 否 正 确 。 

很 多 开发 人 员 都 知道 JUnit, 也 会 在 开发 时 不 定期 用 到 它 。 但 他 们 一 般 是 写 完 和 实现 代码 之 后 才 
编写 测试 代码 ， 因 此 体会 不 到 TDD 的 益处 。 

尽管 TDD 的 概念 看 起 来 非常 普及 , 但 实际 上 很 多 开发 人 员 并 不 清楚 为 什么 要 采用 TDD。 对 于 
很 多 开发 人 员 来 说 ,“ 为 什么 要 写 测试 驱动 代码 以 及 有 什么 好 处 ”一 直 是 个 问题 。 

我 们 认为 消除 恐惧 和 不 确定 性 是 编 与 测试 驱动 代码 的 重要 原因 。Kent Beck( JUnit 测 试 框 架 
的 发 明 人 之 一 ) 在 Test-Driven Development: by Example” ( Addison-Wesley Professional，2002 ) 一 
书 中 对 此 总 结 得 很 好 ， 

口 巧 惧 会 让 你 小 心 试探 ; 

口 恐惧 会 让 你 尽量 减少 沟通 ; 

D 恐惧 会 让 你 羞 于 得 到 反馈 ; 

玖 惧 会 让 你 脾气 暴躁 。 

TDD 可 以 祛除 仙 届 ， 让 优秀 的 Java 开 发 者 变 得 更 加 自信、 善于 沟通 、 乐 于 接受 并 更 加 快乐 。 


OD 中 文 版 《测试 驱动 开发 》 已 由 中 国电 力 出 版 社 于 2004 年 出 版 。 一 一 编者 注 
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换 句 话说 ，TDD 能 帮 你 摆脱 下 面 这 种 心态 ; 

口 在 开始 新 工作 时 ,“ 我 不 知道 从 哪里 开始 ， 所 以 只 好 将 就 着 做 一 些 修 改 ”; 

口 在 修改 已 有 代码 时 ,“ 我 不 知道 现 有 代码 怎么 运行 ， 所 以 我 私下 认为 不 能 磁 它 们 ”。 

TDD 带 来 的 很 多 好 处 并 不 会 马上 显现 ， 

口 更 清晰 的 代码 一 一 只 写 需 要 的 代码 ; 

口 更 好 的 设计 一 一 有 些 开 发 人 员 管 TDD 叫 测试 驱动 的 设计 ; 

口 更 出 色 的 灵活 性 一 -TDD 鼓 励 按 接口 编码 ; 

口 更 快速 的 反馈 一 一 不 会 直到 系统 上 线 才 知道 bug 的 存在 。 

刚 人 门 的 开发 人 员 有 时 认为 TDD 不 是 “普通 ”开发 人 员 用 的 技术 , 这 是 他 们 采用 TDD 的 一 个 
障碍 。 他 们 的 感觉 是 只 有 那些 想象 中 的 “敏捷 派 ” 或 其 他 神秘 组 织 的 成 员 才 会 用 TDD。 这 种 认识 
完全 错误 ， 我 们 会 在 后 面 解 释 。TDD 是 给 所 有 开发 人 员 使 用 的 技术 。 

另外 ， 人 敏捷 和 软件 工艺 运动 都 是 为 了 让 开发 人 员 活 得 更 轻松 。 它 们 肯定 不 会 拒绝 别人 使 用 
TDD 或 其 他 任何 技术 。 

本 章 首 先 解释 TDD 背 后 的 基本 思想 红 一 绿 一 重 构 循 环 ， 然 后 介绍 Java 测 试 框架 中 的 主力 
JUnit， 并 用 一 个 简单 的 例子 来 阐明 其 原则 。 


训 们 自己 其 至 也 捉 护 它 (大 多 数 
要 对 有 用 的 4 未 本。 " IDI 中 光村 


时 候 都 是 如 此 )。 te pe 
7 -发 技术 ， 仅 此 而 Es 


接 下 来 ,我们 会 介绍 TDD 使 用 的 四 大 类 伪装 对 象 。 它 们 能 简化 受 试 代码 和 第 三 方 类 库 中 代码 
的 隅 离 ， 或 隔离 数据 库 之 类 的 子 系统 行为 ， 所 以 它们 很 重要 。 随 着 依赖 项 变 得 越 来 越 复 杂 ， 伪 装 
对 象 也 要 变 得 越 来 越 聪明 。 最 终 我 们 会 介绍 模拟 和 Mockito 类 库 ， 它 是 一 个 流行 的 模拟 工具 ， 可 
以 让 开发 人 员 在 不 受 外 部 系统 影响 的 环境 下 进行 测试 。 

开发 人 员 非 常熟 悉 Java 测 试 框架 ( 特别 是 JUnit ), 并 且 一 般 都 有 用 它们 编写 测试 代码 的 经 验 。 
但 对 于 如 何 用 测试 驱动 Scala、Clojure 等 新 语言 ， 你 可 能 毫 无 头绪 。 因 此 我 们 会 介绍 Scala 测 试 框 
涤 ScalaTest， 以 确保 你 能 在 开发 Scala 代 码 时 应 用 TDD，。 

让 我 们 开始 了 解 这 个 有 点 奇怪 的 TDD 吧 。 


11.1 TDD 概览 
TDD 可 以 应 用 在 多 个 层级 上 。 表 11-1 列 出 了 通常 会 采用 TDD 的 四 个 测试 层级 。 


1 者 
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表 11-1 TDD 的 测试 层级 
层 级 ”“ 描 述 例子 
单元 测试 通过 测试 验证 一 个 类 中 包含 的 代码 测试 BigDecimal 类 中 的 方法 
提成 测试 通过 测试 验证 类 之 间 的 变 玉 测试 curreney 瞻 以 及 它 如 何 腿 BigDpecimal 交 互 
系统 测试 通过 测试 验证 运行 的 系统 从 UI 到 currency 类 测试 会 计 系统 
系统 集成 测试 。” 通过 测试 验证 运行 的 系统 ， 包 括 第 三 方 组件 。 测试 会 计 系统 ， 包 括 它 与 第 三 方 报表 系统 间 的 交互 


在 单元 测试 中 使 用 TDD 是 最 容易 的 ， 如 果 你 对 TDD 不 熟悉 , 这 一 层 就 是 个 很 好 的 起 点 。 本 他 
主要 讲述 如 何在 单元 测试 层 中 使 用 TDD。 后 续 章 节 会 讨论 其 他 层级 , 包括 第 三 方 组 件 和 子 系统 的 
测试 。 


提示 “处理 没有 或 只 有 很 少 测 试 的 遗留 代码 是 小 恶 怖 的 任务 。 我 们 几乎 不 可 能 把 所 有 测试 都 追 
加 上 ， 因 此 ， 应 该 只 是 为 添加 的 新 功能 加 上 测试 代码 。 请 参阅 Michael Feathers 的 Working 
Ejfectively with Legacy Code”( Prentice Hall，2004 ) 获取 更 多 帮助 。 


我 们 一 开始 会 简单 介绍 一 下 TDD 的 基本 前 提 一 一 红 一 绿 一 重 构 循环 一 一 用 JUnit 测 试 计算 剧 
院 门 票 销售 收入 的 代码 ”~。 只 要 遵照 红 一 绿 一 重 构 循环 ， 基 本 上 就 可 以 使 用 TDD! 之 后 我 们 会 探 
究 一 下 红 一 绿 一 重 构 循 环 背 后 的 思想 , 让 你 对 为 什么 应 该 采用 这 种 技术 有 更 清楚 地 认识 。 最 后 我 
们 将 介绍 JUnit 这 个 公认 的 Java 开 发 者 测试 框架 ， 讲 解 它 的 基本 用 法 。 

让 我 们 开始 吧 ， 先 来 一 个 TDD 三 步 ( 红 一 绿 一 重 构 ) 测试 计算 剧院 门票 销售 收入 的 实际 
例子 。 


11.1.1 一 个 测试 用 例 


如 果 你 有 TDD 方 面 的 经 验 ， 可 以 目 行 决 定 是 否 跳 过 这 一 节 ， 不 过 这 个 小 例子 中 有 些 新 东西 。 
假定 有 人 要 你 写 一 个 坚 帮 网 石 的 方法 来 计算 剧院 门票 的 销售 收入。 剧院 会 计 最 初 给 出 的 业务 规则 
很 简单 : 

口 门票 的 底价 是 30 美 元 ; 

口 总 收入 = 售 出 票数 * 价 格 ; 

口 剧院 有 100 个 座位 。 
因为 剧院 工作 人 员 不 慌 软 件 ， 所 以 他 们 现在 还 必须 手工 录入 门票 的 销售 数 

如 果 你 做 过 TDD， 应 该 知道 它 的 三 个 基本 步骤 : 红 、 绿 、 重 构 。 如 果 刚 接 衣 
习 一 下 ， 那 就 请 看 一 下 Kent Beck 在 《测试 驱动 开发 》 中 对 这 些 步 又 的 定义 ， 


量 。 
中 TDD ， 或 者 想 复 


加 中 文 版 《修改 代码 的 艺术 》 已 由 人 民 邮 电 出 版 社 于 2007 年 出 版 (更 多 信息 请 参见 http-wrwwituring com cnyboald536 ) 
一 编者 注 
加 销售 剧院 门票 在 我 的 家 乡 伦敦 是 个 大 生意 ， 最 起 码 在 我 们 写 这 本 书 的 时 候 是 。 


1 入 
和 
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(1) 红 ， 写 一 些 不 能 用 的 测试 代码 ( 失败 测试 ); 
(2) 绿 ， 尽 快 让 测试 通过 (通过 测试 ); 
(3) 重 构 ， 消 除 重 复 (经 过 细 化 的 通过 测试 ) 。 
为 了 让 你 了 解 TicketRevenue 应 该 达到 什么 效果 ， 请 先 看 一 下 这 些 伪 代码 。 
estimateRevenue (int numberofTicketssold) 


if (numberofTicketssold is less than 0 OR greater than 100) 


then 
Deal with error and exit 
到 ] 所 


revenue = 30 * numberoOfTicketssold:; 
return revenue; 
endif 


注意 ,和 干 万 别 太 深 入 。 测 试 最 终 会 驱动 设计 ， 也 会 部 分 影响 实现 。 


壮 意 我们 在 11.1.2 节 会 涉及 开始 失败 测试 的 办 法 ,但 在 这 个 例子 中 我 们 准备 写 一 个 其 至 还 无 法 
编译 的 测试 ! 


接 下 来 我 们 先 用 JUnit 写 一 个 失败 单元 测试 。 如 果 你 不 了 解 JUnit, 请 跳 到 11.1.4 节 , 然后 再 回来 。 

1. 编写 失败 测试 〈 红 ) 

这 一 步 的 要 点 是 以 一 个 会 失败 的 测试 开始 。 实 际 上 ， 这 个 测试 甚至 无 法 编译 ,因为 你 还 没有 
TicketRevenue 类 | 

在 跟 会 计 开 过 一 个 简短 的 白板 会 议 后 , 你 意识 到 测试 代码 需要 覆盖 五 种 情况 : 售票 数量 为 负 
数 、0、1、2~100， 还 有 大 于 100。 


提示 编写 测试 代码 ( 特别 是 牵扯 到 数值 时 ) 有 一 个 很 好 的 经 验 法 则 ， 要 者 虚 值 为 0/null、1 和 
很 多 (NN) 的 情况 。 再 进一步 考虑 N 上 的 其 他 限制 ， 比 如 数量 为 负 或 超出 上 限 。 


我 们 决定 先 写 一 个 测试 种 盖 销 售 一 张 门票 收入 的 情况 。 测 试 代码 看 起 来 应 该 如 代码 清单 11-1 
所 示 ( 记 住 这 个 阶段 不 用 编写 完美 的 通过 测试 )。 


代码 清单 11-1 为 TicketRevenue 编 写 的 失败 单元 
import Java.math.BigDecimal; 
import static junit,framework.Assert .*; 
import org.Junit .Before:; 
import org.jJunit.Test:; 
public class TicketRevenueTest | 


private TicketRevenue venueRevenue; 
private BigDecimal expectedRevenue; 
Betiore 
Public void setUp() 1 

venueRevenue = new TicketRevenue!(); 


} 


1 入 
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a 销售 一 张 票 的 情况 
public void oneTicketSoldIsThirtyInRevenue () | | 


expectedRevenue = new BigDecimal ("30").,; 
assertEquals lexpectedRevenue, venueRevenue.estimateTotalRevenue (1)):; 
| 
| 


测试 期 望 销 售 一 张 门 票 得 到 的 收入 等 于 30。 
但 这 个 测试 不 能 编译 ， 因 为 有 estimateTotalRevenue (int numberOfTicketssold) 方 
法 的 TicketRevenue 类 还 不 存在 呢 。 为 了 运行 测试 , 可 以 先 随 便 写 一 个 让 测试 可 以 编 至 的 实现 。 
public class TicketRevenue | 
public BigDecimal estimateTotalRevenue lint i) { 
: return BigDecimal .2ERO; 
| 
现在 测试 代码 能 编译 了 ， 你 可 以 在 自己 喜欢 的 IDE 中 运行 它 。 每 种 IDE 都 有 目 己 运行 JUnit 测 
试 的 办 法 ， 但 一 般 都 能 在 选中 测试 类 后 ， 从 右键 弹出 菜单 中 选择 运行 测试 。 一 旦 运行 ，IDE- 一 般 
都 会 更 新 窗口 告诉 你 测试 失败 了 ， 因 为 你 所 期 望 的 30 和 estimateTotalRevenue(1) ;返回 的 值 
不 符 ， 它 的 返回 值 是 0。 
失败 测试 有 了 ， 接 下 来 该 做 通过 测试 了 ( 变 绿 )。 
2. 编写 通过 测试 〈 绿 ) 
这 一 步 的 要 点 是 让 测试 通过 ， 但 没 必要 把 实现 做 到 完美 。 给 TicketRevenue 类 一 个 更 好 的 
estimateTotalRevenue 实 现 (不 会 只 返回 0 )， 可 以 让 测试 通过 ( 变 绿 )。 
记 住 ， 这 一 阶段 只 要 让 测试 通过 就 行 ， 没 必要 追求 完美 。 代 码 可 能 如 代码 清单 11-2 所 示 : 


Ee EE ,本 Ee l 3 上 ii 本 1 本 
iB: Fe Bie ph 和 | JW 
站 四 和 = | 人 | 4 i = 要 - 有 
和 | rv | = | | J 中 
7 丙 愉 1 1- 一 由 入 HYJT CRELnEe 
1 1 有 是 下 er Fr "ss 1 | 让 \ - wd 站 十 多- 本 si Wr Bl md 村 T hae} 瑟 


import jJava.math.BigDecimal:; 


public class TicketRevenue | 


Public BigDecimal estimateTotalRevenue (int numberoOfTicketasSold) | 
BigDecimal totalRevenue = BigDecimal .ZERO, 
if (numberofTicketsSold == 1) | 


totalRevenue = new BigDecimal ("30"); , 
} | 通过 测试 的 实现 


return totalRevenue,; 
} 
) 
现在 再 运行 测试 ， 通 过 了 ! 而 且 在 大 多 数 IDE 中 ， 会 用 一 个 绿 条 或 对 名 来 表示 测试 通过 。 图 
11-1 是 在 Eclipse 中 通过 测试 的 界面 。 
接 下 来 的 问题 是 你 能 不 能 说 “我 摘 定 了 ”, 然后 去 做 下 一 项 工作 ? 我们 可 以 负责 任 地 告诉 你 : 
不 是 ! ”你 会 妇 不 住 想 完善 前 面 的 代码 ， 那 现在 我 们 就 开始 吧 。 


1 入 
Yr 


总 入 和 1 7 vp i Par EC [eu ja chips a i | 和 i 
Hr | 光 。 0- “人 Ex Pryy? pd [ee -i 
> Nengater 二 生生 | wm Trieevenw Tete 所 = 

一 rag Ch jp chepterll Visting 1 于 | 


or on, sth, Bi ge imal ;| 
le elous Thichathleweletl 站 


| pe 下 下 下 二 三 由 网页 刘 二 FE 站。 可 并 证 古 者 机 放量 曾 Fi 过 
| riveote Bligetieol Wnsecttdesr: 


obite -id 和 UL | 
sure = bn chmteer 
} 


a FE pee iT tylnieve mj 下 
放量 昌 请 导 pi ms 重生 痢 打 于 丰 让 二 汪 直 让 有 下 
ET ph 国有 帮工 征 因 呈 加 EAE ， mje 允 中 Ede 


改 Fabmms | @ jwasoc 口 区 Decaraton | 于 Tusk tt | Oudine 


| 
i 
和 省 


BR We 


图 11-1 Bclipse DE 中 表示 测试 通过 的 归 条 ， 纸 质 版 印刷 出 来 是 中 度 灰 色 


3. 重 构 测试 
这 一 步 的 要 点 是 看 看 为 了 通过 测试 写 的 快速 实现 ,确保 你 遵循 了 通行 的 惯例 。 代 码 清单 11-2 
中 的 代码 明显 可 以 更 清晰 、 更 整洁 。 你 肯定 要 重 构 ， 以 减轻 自己 和 他 人 的 技术 债务 。 


技术 债务 Ward Cunningham 发 明 的 说 法 , 指 我 们 现在 临时 资 合 出 来 的 设计 或 代码 将 来 会 让 我 们 
付出 更 多 的 成 本 (工作 小 


记 住 。 有 了 通过 测试 ， 可 以 放心 大 胆 地 重 构 。 应 该 实现 的 业务 逻辑 不 可 能 会 被 忽视 。 


提示 “编写 最 初 的 通过 测试 代码 的 另 一 个 好 处 是 开发 进度 可 以 更 快 。 团 队 中 的 其 他 人 可 以 马上 
用 第 一 版 代码 跟 更 大 的 代码 库 一 起 测试 ( 集成 测试 及 更 大 范围 的 测试 )。 


在 代码 清单 11-3 中 ， 我 们 不 想 再 用 魔法 数字 了 一 一 要 让 票 价 (30 ) 出 现在 代码 中 。 
代码 清单 11-3 ”通过 汶 


import java.math. Dunit , 


1 入 
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public class TicketRevenue | i 
不 用 矿 法 数字 了 
private final static int TICKET PRICE = 30; = | 


public BigDecimal estimateTotalRevenue (int numberoOfTicketsso1ld) | 
BigDecimal totalRevenue = BigDecimal .ZERO;} 
if (numberOfTicketsSold == 1) | 
totalRevenue = 
new BigDecimal (TICKET PRICE * 


numberofTicketsSsold).; : 
| | 


return totalRevenue: 
| 
经 过 这 次 重 构 ， 代 码 得 到 了 改善 ， 但 很 明显 它 还 设 有 涵盖 所 有 情况 ( 售票 数量 为 负 值 、0、 
2~100 和 大 于 100 )。 你 不 能 只 是 拼命 地 狂 其 他 情况 下 的 实现 应 该 是 什么 样 ， 而 应 该 做 更 多 测试 驱 
动 的 设计 和 实现 。 下 一 节 会 继续 按照 测试 驱动 设计 的 方式 ,市 你 看 更 多 的 测试 用 例 。 


11.1.2 多 个 测试 用 例 


按照 TDD 风 格 ， 应 该 继续 为 门票 销售 数量 为 负 值 、0、2~100 和 大 于 100 的 情况 依次 添加 测试 
用 例 。 但 还 有 一 种 办 法 ， 一 次 写 一 组 测试 用 例 也 行 ， 特 别 是 在 它们 跟 最 初 的 测试 有 关 的 时 候 。 

注意 , 这 次 仍然 要 遵循 红 一 绿 一 重 构 的 循环 周期 。 在 把 这 些 用 例 都 加 上 之 后 ,你 应 该 会 得 到 
一 个 带 有 失败 测试 ( 红 ) 的 测试 类 ， 如 代码 清单 11-4 所 示 。 


代码 清单 11-4 TicketRevenue 的 失败 单元 测试 
import java.math.BigDecimal; 
import static Junit.framework.Assert,™*; 
import org.junit.Test,; 
Public class TicketRevenueTeat | 


private TicketRevenue venueRevenue,; 
private BigDecimal expectedRevenue,; 


Before 
public void setUp() | 
VenueRevenue = New TicketRevenue |().; 


| 


STest (expected=IllegalArgument Exception.clasas) 销量 为 负 什 
Publiec void failIlfLessThanZeroTicketsAreSold(}) | 

venueRevenue .estimateTotalRevenue(-=-1):; 
} | 
多 Test 销量 为 0 
public void zerosalesEqualszZeroRevenue() { 

assertEqguals (BigDecimal .ZERO, venueRevenue. ER to 


} 


呈 Test 销量 为 1 
public void oneTicketSoldIsThirtyInRevenue() | RE 
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expectedRevenue = new BigDecimal ("30"); 
assertEquals (expectedRevenue, venueRevenue.estimateTotalRevenue (1)); 


| 


Test ] 销量 为 N 
public void tenTicketsSoldIseThreeHundredIinRevenue(}) | \ 
expectedRevenue = new BigDecimal ("300"); 


assertEquals (expectedRevenue, venueRevenue .eastimateTotalRevenue (10)),; 


} 

HTest (expected=IllegalArgument Exception.class) 

public void failIfMoreThanoneHundredTicketsAreSold() 1 | 
venueRevenue .estimateTotalRevenue (101); 销量 大 于 1 

| 


) 
为 通过 所 有 测试 ( 绿 ) 写 的 基本 实现 版 看 起 来 应 该 如 代码 清单 11-5 所 示 。 


’ JJ 可 让 式 的 第 一 版 ricketRevenue 
Fe java. ey BigDecimal; 


public class TicketRevenue | 
Public BigDecimal estimateTotalRevaenue {int numberOfTicketsSold) 
throws IllegalArgqumentException | 
BigDecimal totalRevenue = null:; 
if (numberOfTicketssold < 0}) 1 
throw new lllegalArgumentException ("Must be > -1"); 


| 
if {numberOfTicketsSold == 0) 1 

totalRevenue = BigDecimal .ZERO; 
| 


if (numberofTicketsSold == 1) | 
totalRevenue = new BigDecimal ("30").; 
} 


if (numberOfTicketaSold == 101) | 
throw new IllegalArgumentException("Must be < 101").; 
} 


else | 
totalRevenue = 


new BigDecimal (30 * numberOfTicketsSold).; 
} | 销量 为 


return totalRevenue:; 


有 了 刚刚 完成 的 实现 ， 现 在 你 的 测试 就 变 成 通过 测试 了 。 

按照 TDD 循 环 周期 , 现在 该 重 构 这 个 实现 了 。 比 如 说 , 可 以 把 不 合法 的 numberofTicketssold 
情况 (人 负数 或 者 大 于 100 ) 放 到 一 个 i1f 夺 名 中 , 并 用 公式 (TICKET PRICE * numberOfTicketsSold) 
返回 所 有 合法 numberoOfTicketsSol9 的 收入 。 人 代码 清 单 11-6 应 该 跟 重 构 之 后 的 代码 很 像 。 


代码 清单 11-6 


import java.math.BigDecimal; 


异常 情况 
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public class TicketRevenue | 
private final static int TICKET PRICE = 30; 


public BigDecimal estimateTotalRevenue (int numberOftTicketsSsold) 
throws IllegalArgumentException { 


if (numberofTicketsSsold < 0 || numberofTicketsSold > 100) | 
throw new IllegalArgumentException 


("# Tix sold must == 1..100"),; 
! | 异常 状况 


return new BigDecimal 


(TICKET PRICE * numberOfTicketsSold); ; : 
} 所 有 其 他 情况 
| 


新 的 TicketRevenue 类 更 加 紧凑 ， 并 且 还 通过 了 所 有 测试 ! 现在 你 已 经 完成 了 整个 红 一 绿 
一 重 构 循 环 ， 可 以 信心 满 满 地 开始 实现 下 一 个 业务 逻辑 了 。 另 外 ， 如 果 你 (或 会 计 ) 发 现 漏 掉 了 
任何 边界 情况 ， 比 如 有 浮动 标价， 也 可 以 再 次 开始 一 个 循环 。 

我 们 强烈 建议 你 弄 明 白 红 一 绿 一 重 构 的 TDD 方 式 背 后 的 原理 , 也 就 是 我 们 接 下 来 要 讨论 的 内 
容 。 但 如 果 你 没什么 耐心 ， 可 以 直接 跳 到 11.1.4 节 学 习 JUnit， 或 11.2 节 了 解 用 来 测试 第 三 方 代码 
的 测试 替身 。 : 
11.1.3 ”深入 思考 红 一 绿 一 重 构 循 环 

这 一 节 会 在 前 面 例子 的 基础 上 探索 TDD 背 后 的 一 些 思 想 。 我 们 会 再 次 谈论 红 一 绿 一 重 构 循 
环 ， 你 应 该 还 记得 第 一 步 是 写 失 败 测 试 。 但 这 也 有 几 种 不 同 的 方式 。 

1. 失败 测试 ( 红 ) 

一 些 开 发 人 员 真 的 喜欢 编写 编译 失败 的 测试 ,喜欢 等 到 绿色 步骤 才 提 供 实现 代码 。 也 有 一 些 
开发 人 员 喜 欢 先 把 测试 调用 的 方法 存根 写 出 来 ,这样 虽 然 测 试 代 码 能 编译 ,但 还 是 会 失败 。 我 们 
党 得 怎么 样 都 行 ， 随 意 就 好 。 


提示 ”这 些 测 试 代 码 是 实现 的 第 一 个 客户 ， 所 以 应 该 认真 考虑 该 怎么 设计 它们 ， 方法 定义 看 起 
来 应 该 是 什么 样 的 。 还 应 访问 自己 几 个 问题 ; 该 传 什么 参数 进去 ? 期 望 的 返回 值 是 什么 ? 
会 不 会 有 异常 情况 ? 另外 ,不 要 忘 了 测试 重要 领域 对 后 的 eauals() 和 hashcode1() 方 法 。 


一 日 写 完 失 败 测 试 ， 就 该 进入 下 一 阶段 了 : 让 它 通过 。 

2, 通过 测试 ( 绿 ) 

这 一 步 应 该 尽量 少 写 代码 ， 只 要 保证 测试 通过 就 行 。 也 就 是 说 你 不 用 把 实现 做 到 完美 那 是 
重 构 阶段 的 工作 。 

测试 通过 之 后 , 你 就 可 以 告诉 同事 ,你 的 代码 已 经 实现 了 它 应 该 实现 的 功能 ， 他们 可 以 拿 去 
用 了 。 

3. 重 构 

在 这 一 步 中 应 该 重 构 实 现代 码 。 可 以 重 构 的 地 方 数不胜数 , 但 有 几 个 应 该 重点 关注 的 ， 比 如 
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去 掉 硬 编码 的 变量 或 把 大 方法 拆 分 开 。 如 果 是 面向 对 象 的 代码 ， 则 应 该 遵循 S 
SOLID 原 则 是 Bob 大 叔 ( Robert Martin ) 提出 来 的 ,请 参见 表 11-2。 要 了 解 更 详细 的 信息 ， 可 以 
参考 他 的 文章 “The Principles of OOD™( http://butunclebob.com/ArticleS.UncleBob.PrinciplesOfOod )。 


表 11-2 面向 对 象 代码 的 SOLID 原 则 


原 ” 则 描 述 | 
单一 职责 原则 (SRP) ”每 个 对 象 都 应 该 做 一 件 事 ， 并 且 只 做 一 件 事 
开放 /封闭 原则 ( OCP ) 对 象 应 该 是 可 扩展 、 但 不 可 修改 的 
里 氏 替 换 原 则 (LSP ) 对 象 应 该 可 以 被 它 的 子 类 型 实例 替换 
接口 隔离 原则 (1SP ) 特定 的 小 接口 更 好 
依赖 倒置 原则 ( DIP ) 不 要 依赖 具体 实现 ( 请 参见 第 3 章 关 于 依赖 注 人 的 内 容 ) 


提示 我们 还 要 向 你 推荐 Checkstyle 和 FindBugs 这 两 个 静态 代码 分 析 工 具 ( 第 12 章 还 有 更 多 )。 
Joshua Bloch 的 Efective Java, Sencond Edition"( Addison-Wesley, 2008 ) 也 是 好 责 源 ， 其 中 
有 很 多 Java 语 言 的 技巧 和 穿 门 。 


测试 代码 本 身 的 重 构 是 个 容易 被 人 遗忘 的 和 角落。 你 可 以 把 通用 的 设置 和 拆 外 代码 提取 出 来 ， 
可 以 重 命名 测试 以 更 准确 地 反应 它 的 测试 意图 ,还 可 以 根据 静态 分 析 工具 的 分 析 结 果 做 些小 修订 。 

现在 你 已 经 能 跟 上 TDD 的 三 步 走 了 ,该 去 熟悉 一 下 JUnit 了 , 它 可 是 在 Java 里 写 TDD 代 码 的 默 
认 工 具 。 


11.1.4 JUnit 


JUnit 是 公认 的 Java 项 目测 试 框架 。 当 然 , 除了 JUnit 还 有 其 他 测试 框架 ,比如 拥有 不 少 和 追随 者 
的 TestNG，,， 但 目前 JUnit 下 是 Java 测 试 界 的 主流 。 


注意 ”如 果 你 熟悉 JUnit， 可 以 跳 到 11.2 节 。 


JUnit 有 三 个 主要 特性 : 

口 用 于 测试 预期 结果 和 异常 的 断言 ， 比 如 assertEquals |(); 

D 设置 和 拆 外 通用 测试 数据 的 能 力 ， 比 如 eBefore 和 @after:; 

口 运行 测试 套件 的 测试 运行 器 。 

JUnit 用 简单 的 注解 模型 提供 了 很 多 重要 的 功能 。 

大 多 数 IDE ( 比如 Eclipse、IntelliJ 和 NetBeans ) 都 内 置 了 JUnit， 如 果 你 用 的 正好 是 其 中 之 一 ， 
就 不 用 目 己 去 下 载 、 安 装 或 配置 JUnit 了。 如 果 你 的 IDE 没 有 安装 JUnit， 可 以 访问 www.junit.org 查 


山 《Effective Java 中 文 版 》 由 机 械 工 业 出 版 社 于 2003 年 出 版 。 一 一 编者 注 


只 自 在 读书 @ 
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看 它 的 下 载 和 安装 指导 =。 
注意 我 们 用 的 是 JUnit 4.8.2。 如 果 你 要 练习 本 章 中 的 例子 ， 建 议 也 用 这 个 版 本 。 


一 个 基本 的 JUnit 测 试 包 含 下 面 这 些 元 系 : 

用 8Before 标 记 设 置 方 法 ， 在 每 个 测试 运行 前 准备 测试 数据 ; 

口 用 eaftez 标 记 拆 印 方 法 ， 在 每 个 训 试 运行 完成 后 拆 鲫 测试 数据 ; 

口 测试 方法 本 身 ( 用 emrest 注 解 标记 )。 

为 了 多 了 解 一 下 上 面 这 些 元 素 ， 我 们 来 看 几 个 非常 基本 的 JUnit 测 试 。 

比如 OpemJDK 团 队 要 你 给 Bigpecimal 类 写 个 单元 测试 。 第 一 个 测试 是 检查 加 法 (1.5 + 1.5==3. 
0) ; ， 第 二 个 测试 是 检查 用 非 数字 值 创 建 Bigpecimal 实 例 时 会 抛 出 NumberFormatExcepticn 异 常 。 


注意 ”我 们 在 本 章 的 例子 中 经 常 同时 给 出 多 个 失败 测试 ， 实 现 ( 绿 ) 和 重 构 。 这 违背 了 纯粹 的 
TDD 单 个 测试 贯 作 红 一 绿 一 重 构 循 环 的 原则 ， 但 孝 可 以 让 我 们 在 本 章 中 放 入 更 多 例子 。 
不 过 在 你 编码 时 ， 应 该 尽 可 能 地 遵守 单个 测试 循环 的 开发 模型 。 


要 运行 代码 清单 11-7， 可 以 在 IDE 里 的 源码 文件 上 点 击 右键 ， 选 择 运 行 或 测试 选项 ( 三 个 主 
流 IDE 中 都 有 显眼 的 Run Test 或 Run File 选 项 )。 


import java.math.BigDecimal,; 


import org.jJunit.*; a Ee 
import estatic org. unit .Assert.*; | 标准 的 JUnit 导 入 


public class BigDecimalTest I 
private BigDecimal x; 


SBbefore z 各 每 个 测试 之 前 的 设置 
public void setUp() { x = new BigDecimal("1.5"); | 


re 2 每 个 测试 之 后 的 拆 外 
public void tearDown{) { x = null; } ] 


矿工 总 豆 七 
public void addingTwoBigDecimals() | ae 执行 测试 
| 


assertEquals(lnew BigDecimal ("3.0"), x.add (x)); 
} 


STest (expected=NumberFormat Exception.class) . 
public void numberFormatExceptionIfNotANumber{() | oO 处 理 章 料 中 的 异常 
xX = new BigDecimal ("Not a number"):; \ : i 
| 
| 


DD 第 12 章 会 讲 到 JUnit 和 Maven 的 集成 。 
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在 每 个 测试 运行 之 前 ，x 在 eefore 区 域 中 被 设置 为 Bigpecimal1("1.5")@@。 这 会 确保 每 
个 测试 处 理 的 都 是 已 知 值 x， 而 不 是 被 之 前 运行 的 测试 修改 过 的 中 间 值 。 在 每 个 测试 运行 之 后 ， 
在 eafter 区 域 中 确保 zx 被 设 为 nul1 和 (以 便 x 可 以 被 垃圾 收集 ), 然后 用 assertEquals() (JUnit 
众多 静态 assertX 方 法 之 一 ) 测试 pigDecimal .ada|() 的 返回 结果 是 否 符 合 期 望 估 , 为 了 处 理 预 
期 的 异常 ， 在 @Test 上 加 上 了 可 选 的 expected 参 数 @， 

进入 TDD 最 佳 状态 的 最 好 办 法 就 是 动手 实践 。 把 TDD 厚 则 牢 牢 印 在 你 的 脑海 里 ， 把 JUnit 框 
架 搞 明日 ， 你 就 可 以 开始 了 ! 通过 这 些 例子 你 也 能 看 出 来 ， 单 元 测试 级 的 TDD 很 容易 掌握 。 

但 所 有 TDD 从 业者 最 终 都 要 测试 使 用 依赖 项 或 子 系统 的 代码 .下 一 节 就 会 讲 到 那些 代码 的 测 
试 技术 。 


11.2 测试 替身 


如 果 你 继续 用 TDD 风 格 编码 ,很 快 就 会 遇 到 需要 引用 ( 经 常 是 第 三 方 的 ) 依赖 项 或 子 系统 的 
情况 。 在 这 种 情况 下 ,你 肯定 想 把 测试 代码 跟 依赖 项 隔离 开 ， 以 保证 测试 代码 仅仅 针对 于 实际 构 
建 的 代码 , 你 肯定 还 想 让 测试 代码 尽 可 能 快速 运行 。 而 调用 第 三 方 依赖 项 或 子 系统 ( 比如 数据 库 ) 
可 能 会 花 很 长 时 间 ， 也 就 是 说 会 背 失 TDD 快 速 啊 应 的 优势 ( 在 单元 测试 层面 尤其 如 此 )。 测 试 替 
身 (test double ) 就 是 为 解决 这 个 问题 而 生 的 。 

你 在 这 一 节 将 学 会 如 何 用 测试 替身 有 效 隔离 依赖 项 和 子 系统 ,看 到 使 用 四 种 测试 替身 ( 虚设 、 
伪装 、 存 根 和 模拟 ) 的 例子 。 

在 最 复杂 的 情况 下 ， 也 就 是 测试 有 外 部 依赖 项 ( 比如 分 布 式 服务 或 网 络 服务 ) 的 代码 时 ， 依 
不 注 人 技术 ( 见 第 3 章 ) 会 和 测试 蔡 身 联手 来 瓜 救 你 ， 即 便 是 看 上 去 大 得 吓人 的 系统 ， 它 们 也 能 
保 你 安全 无 不 。 


和 Te et DI 框架 的 参考 实现 。 阅读 这 一 节 
时 你 很 可 能 边 看 边 想 ;, “他 们 怎么 不 用 Guice 呢 ? ” 


简 言 之 ， 对 于 这 些 代 码 ， 即 便 引 入 像 Guice 这 样 简单 的 框架 都 显得 过 于 复杂 。 记 住 ，DI 是 
一 项 技术 。 不 要 纯 柠 为 了 使 用 框架 而 使 用 它 。 


Gerard Meszaros 在 他 的 xUnit Test Patterns”( Addison-Wesley Professional，2007 ) 一 书 中 给 出 
了 测试 符 刁 的 简单 解释 ， 我 们 很 染 幸 能 在 这 里 引用 他 的 说 法 :“ 测 试 替身 ( 想 一 想 特技 演员 ) 泛 
指 任 何 出 于 测试 目的 替换 真实 对 象 的 假冒 对 象 .” 

Meszaros 接 着 定义 了 四 种 测试 蔡司 ， 如 表 11-3 所 示 。 

里 然 看 起 来 很 抽象 , 但 见 到 例子 你 就 知道 了 ,它们 非常 容易 理解 。 让 我 们 先 从 虚设 对 象 开始 
淖 起 。 


DD 本 书 中 文 版 《xUnit 测 试 模式 ， 测试 码 重 构 》 已 由 清华 大 学 出 版 社 于 2009 年 出 版 。 一 一 译 者 注 
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表 11-3 ”四 种 测试 替身 
类 型 on. 描述 
虚设 兰 身 只 传递 不 使 用 的 对 象 。 一 般 用 于 填充 方法 的 参数 列表 
存根 赤身 总 是 返回 相同 预 设 响应 的 对 象 ， 其 中 可 能 也 有 些 虚设 状态 
伪装 替身 可 以 取代 真实 版 本 的 可 用 版 本 ( 当然 在 品质 和 配置 上 达 不 到 生产 环境 要 求 的 标准 ) 
模拟 替身 可 以 表示 一 系列 期 望 值 的 对 象 ， 并 且 可 以 提供 预 设 响应 


11.2.1 虚设 对 象 


在 这 四 种 测试 替身 里 ， 虚 设 对 象 用 起 来 最 容易 。 记 住 , 它 是 用 来 填充 参数 列表 ,或 者 填补 屠 
些 总 也 不 会 用 的 必 填 域 。 大 多 数 情况 下 ， 你 其 至 可 以 传人 一 个 空 对 和 象 或 null。 

我 们 回 到 剧院 门票 者 个 例子 中 。 能 估算 出 一 个 售 周 肥 带 来 的 收入 非常 好 , 但 剧院 老板 考虑 得 
更 长 远 。 千 出 门票 和 预期 收入 的 模型 要 做 得 更 好 ， 并 且 你 还 听 到 有 人 把 忽 : 随 着 需求 增多 ， 系 统 
越 来 越 复 杂 了 。 

你 接 到 一 项 任务 ， 要 对 售 出 票 进行 跟踪 ， 并且 某 些 票 可 以 打 9 折 。 看 起 来 你 需要 一 个 带 有 价 
格 打折 方法 的 Ticket 类。 你 又 从 TDD 循 环 的 失败 测试 开始 了 ， 测 试 重点 是 新 的 getDiscount- 
Price() 方 法 。 你 趣 氨 还 需要 两 个 构造 方法 : 一 个 用 于 常规 价格 的 门票 ,一 个 用 于 可 能 会 打折 的 
门票 。Ticket 对 象 最 终 需 要 两 个 参数 

口 客 己 姓名， 测试 中 绝 不 会 用 到 的 string:; 

口 正常 价格 ,测试 中 会 用 到 的 BigDecimal。 

你 非常 确定 getDiscountPrice() 方 法 肯定 不 会 引用 客户 姓名 ， 也 就 是 说 可 以 给 构造 方法 
传人 一 个 虚设 对 象 ( 我 们 用 的 是 固定 字符 串 "Riley" )， 如 代码 清单 11-8 所 示 。 
代码 清单 11-8 用 虚设 对 象 实 

import org.junit .TeSBt; 


import Java.math.BligDecimal; 
import static org.jJunit .Assert ,*; 


现 的 TicketTest 


public class TicketTest | 


创建 虚设 对 象 


Test 
PUblic void tenPercentDiscount () { 
String dummyMName = "Riley",; -一 
Ticket ticket = new Ticket (dummyName, 传 入 虚设 对 象 
New 


BigDecimal ("10")); 
assertEqguals (new BigDecimal ("9.0"), ticket .getDiascountPrice()):; 


} 
看 到 了 吧 ， 虚 设 对 象 的 概念 很 平常 。 
为 了 让 你 彻底 明白 这 个 概念 ， 我 们 在 代码 清单 11-9 中 给 出 了 部 分 实现 的 micket 类 。 
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et i 和 和 Ps 


public class Ticket | | 默认 价格 
public static final int BASIC TICKET PRICE = 30; 
private static final BigDecimal DISCOUNT RATE = 
new BigDecimal ("0.9").; OO 
| 默认 折扣 


private tinal BigDecimal price:; 
private final String clientName:; 


public Ticket (String clientName) 1 
this,clientName = ClientName:; 
price = new BigDecimal (BASIC TICKET PRICBEI1 :; 


public Ticket (String clientName, BigDecimal price) |{ 
this.clientName = cllientName; 
this.price = price; 

| 

public BigDecimal 9etpricel() | 
return price; 


public BigDecimal getDiscountPrice() | 
return price.multiply (DISCOUNT RRTE) ; 
| 
| 
有 些 开 ReedAGeeS 虚设 对 多 非常 直接 ,， 它 
们 就 是 过 去 为 了 避 倪 出 现 Nul1lPointerException 的 古老 对 象 ， 只 是 为 了 让 代码 能 跑 起 来 


我 们 转 人 下 一 个 测试 赫 身 的 讨论 吧 。 存 根 对 象 (从 复杂 度 来 讲 ) 向 前 迈 出 了 一 步 。 


11.2.2 存根 对 象 


在 使 用 能 够 做 出 相同 响应 的 对 象 代替 真实 实现 的 情况 下 , 就 会 用 到 存根 对 象 。 让 我 们 回 到 剧 
院 门 票 价格 的 例子 中 ， 看 一 下 实际 应 用 。 

写 完 Ticket 类 后 ， 领导 给 你 放 了 个 假 。 你 度 完 假 刚 回来 ， 打 开 邮 箱 就 看 到 一 个 bug 单 ， 报 告 
说 代码 清单 11-8 中 的 tenPercentDiscount() 测试 时 好 时 坏 。 你 一 检查 代码 库 ， 发 现 
tenpercentDiscount () 已 经 被 改 掉 了 。 现 在 新 写 了 一 个 Erice 接 口 ， 而 Ticket 实例 是 由 该 接 
口 的 实现 类 HttpPrice 倒 建 的 。 

经 过 调查 ， 你 又 发 现 一 些 变 化 ， 为 了 从 一 个 外 部 网 站 上 的 第 三 方 类 Ht tpPricingsService 
获得 最 初 的 价格 ， 要 调用 HttpPrice 的 getInitialPrice() 方 法 。 

因此 每 次 调用 getInitialPrice() 都 会 返回 不 同 的 价格 ,。 此外, 它 时 好 时 坏 还 有 几 个 原因 ， 
有 了 时 是 公司 防火 墙 规则 变 了 ， 有 时 是 第 三 方 网 站 无 法 访问 了。 

所 以 测试 就 失败 了 ， 测 试 的 目的 也 不 垃 受 到 了 污染 。 记 住 ， 你 所 要 的 单元 测试 只 是 针对 打 9 
折 的 价格 。 
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注意 涉及 第 三 方 价格 网 站 调用 的 情景 肯定 超出 了 测试 的 责任 范围 。 但 你 可 以 考虑 做 一 个 单独 
竹 盖 Httpprice 类 和 第 三 方 的 HttpPricingService 的 系统 集成 测试 。 


在 用 存根 替换 HttpPrice 类 之 前 ， 先 看 一 下 代码 的 当前 状态 ， 如 下 面 三 段 代 码 ( 代码 清单 
11-10 至 代码 清单 11-12 )。 除 了 跟 Price 接 口 有 关 的 修改 ， 剧院 老板 的 想法 也 变 了 ,觉得 没 必 要 记 
录 是 谁 买 了 标 ， 代码 如 下 所 示 。 


代码 清单 11-10 ”实现 了 新 需求 的 TicketTest 
import org.JjJunit .Test:; 
import java.math.BigDecimal,; 
import static org.junit,Assert .*; 


public class TicketTest | 
实现 了 Price 的 
Na | _ HttpPrice 
public void tenPercentDiscount() | 创建 Ticket 
Price price = new HttpPrice!(),; 
Ticket ticket = new Ticket (price); 
assertEquals (new BigDecimal ("9.0"), 


: ticket getDiscountPrice())}).; L 
| 测试 可 能 会 失败 
| 


下 面 是 新 的 Ticket, 现在 它 包 括 了 一 个 私有 类 FixedPrice, 用 来 处 理 价格 已 知 并 固定 的 情 
况 ， 即 不 需要 从 外 部 源 中 获取 这 些 信息 。 


代码 漠 单 44-41 实现 于 业 漠 求 的 DicReE | 作 本 站 放生 有 村 站 | 下 


import java.math.BigDecimal; 


public class Ticket | 
Public static final int BASIC TICKET PRICE = 30; 
private final Price priceSource: 
private BigDecimal faceValue'= null:; 
private final BigDecimal discountRate:; 


private final class FixedPrice implements Price 1 
public BigDecimal getInitialPrice() { 
return new BigDecimal (BASIC TICKET PRICE); 
| 


| 

public Ticket() { 
priceSource = new Fixedpricel), 
discountRate = new BigDecimal ("1.0"); 

| 

public Ticket (Price price) | 
priceSource = price! 


修改 过 的 构造 方法 


discountRate = new BigDecimal ("1.0"); 


| 
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public Ticket (Price price, ] 修改 过 的 构造 方法 
BigDecimal specialDiscountRate) ( , 


priceSource = price:; 
discountRate = specialDiscountRate:; 


| 


public BigDecimal getDiscountPrice() | 


if (faceVvalue == null) { ei 
faceValue = priceSource.getIinitialPpricel(); 方法 调用 


| 


return faceVvalue .multiply(discountRate); -一 一 
计算 没 变 化 


rr 
5. a 


代码 清单 11-12 ”Price 接 口 及 其 实现 HttpPrice 


import java.math.BigDecimal; 


public interface Price | 
BigDecimal getInitialPricel(); 
} 


public class HttpPrice implements Price | 


Override 加 ne 
public BigDecimal getInitialPricel) | 返回 结果 随机 
return HttpPricingService.getIinitialPricel); 


} 
| 


那么 ， 怎 么 才能 做 出 跟 HttpPricingsService 一 样 的 响应 ”关键 是 想 清 楚 测 试 的 真实 意图 是 
什么 ”在 这 个 例子 中 ,你 要 测 的 是 Ticket 类 中 getDiscountPrice() 方 法 所 做 的 乘法 跟 预 期 一 致 。 

因此 你 可 以 用 总 是 返回 同一 价格 的 存根 StubPrice 换 掉 HttpPrice 类 ,以 调用 getInitial- 
Price()。 这 样 就 可 以 把 价格 经 常 变化 且 时 好 时 坏 的 HttpPrice 类 从 测试 中 隔离 出 去 了 , 使 用 代 
码 清 单 11-13 中 的 实现 ， 测 试 就 可 以 通过 了 。 


代码 清单 11-13 使 用 存根 对 象 的 Ticke i ( 


import org.junit.Test; 
import java.math.BigDecimal; 
import static Grg.Junit .Assert.*; 


public class TicketTest 1 


Test 


public void tenPpercentDiscount() | ] stubprice 存 根 
Price price = new Stubprice'l); 3 


Ticket ticket = new Ticket (pricel); 
aSSertEquals(9.0, | 创建 mcket 


ticket .getDiscountPricel) doubleValue1) ， 


0.0001); 
| ] 检查 价格 


StubPrice 是 个 简单 的 小 类 ， 返 回 的 初始 价格 总 是 10， 如 代码 清单 11-14 所 示 。 
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代码 清单 11-14 ”存根 stubPrice 
import java.math.BigDecimal; 
public class Stubprice implements Price | 


BOverride 


public BigDecimal getInitialPrice() | | 返回 同一 价格 
return new BligDecimal ("i0"i); L- . 
] 


本 
味 ! 现在 测试 又 能 通过 了 ， 重 要 的 是 你 又 可 以 上 毫 不 长 惧 地 重 构 剩 下 的 实现 细 方 了 。 
存根 是 种 挺 实 用 的 测试 替身 , 但 有 时候 我 们 会 希望 存根 所 做 的 工作 可 以 尽 可 能 地 接近 生产 系 


统 ， 这 时 可 以 用 伪装 替身 。 


11.2.3 ”伪装 替身 


伪装 对 象 可 以 看 做 是 存根 的 升级 , 它 所 做 的 工作 几乎 和 生产 代码 一 样 , 但 为 了 满足 测试 需求 
会 走 些 捷径 。 如 果 你 想 让 代码 的 运行 时 环境 非常 接近 生产 环境 ( 连接 真实 的 第 三 方 子 系统 或 依 环 
项 )， 伪 装 替身 特别 有 用 ， 

大 部 分 Java 开 发 人 员 述 早 都 要 编写 跟 数 据 库 交 互 的 代码 ， 特 别 是 在 Java 对 象 上 执行 CRUD 操 
作 。 在 DAO ( Data Access Object， 数 据 访 问 对 和 象 ) 代码 跟 生 产 数据 库 连接 之 前 ,证 明 其 可 用 的 工 
作 通 常会 留 到 系统 集成 测试 阶段 ， 或 者 根本 就 不 做 检查 ! 如 果 能 在 单元 测试 或 集成 测试 阶段 对 
DAO 代 码 进 行 检 查 ， 那 将 会 有 很 多 好 处， 最 重要 的 是 你 能 快速 啊 应 。 

在 这 种 情况 下 可 以 用 伪装 对 象 : 用 来 代表 跟 你 交互 的 数据 库 。 但 自己 写 一 个 代表 数据 库 的 伪 
闭 对 和 象 相 当 困 难 ! 好 在 经 过 数 年 的 演进 ,内 存 数据 库 的 轻巧 易 用 已 经 足以 胜任 这 一 工作 ,HSQLDB 
( www.hsqldb.org ) 是 广泛 用 于 这 一 用 途 的 内 存 数 据 库 。 


i 库 中 ， 以 便 后 期 获取 。Java 


剧院 门票 应 用 进展 良好 ， 下 一 阶段 的 工作 就 是 把 门票 保存 在 数 拓 
中 最 常用 的 数据 库 持久 化 框架 是 Hibernate ( www.hibemate.org )。 


如 果 你 不 了 解 Hibemate 或 HSQLDB， Er re Hit Tn ie 是 一 个 对 象 关系 映射 (ORM) 
框架 , 实现 了 Java 持 久 化 API( JPA ) 标 准 ,。 简 而 言 之 , 你 可 以 调用 简单 的 save、 load、 upds 
还 有 很 多 其 他 的 Java 方 法 来 执行 CRUD 操 作 。 这 和 用 原始 的 SQL 和 JDBC 不 同 ， 并 且 它 经 过 抽象 
隔离 了 特定 数据 库 的 语法 和 语义 。 

HSQLDB 只 是 个 Java 内 存 数 据 库 。 只 要 把 hsqldb.jar 放 到 你 的 CLASSPATH 下 就 可 以 用 了 。 
尽管 在 关闭 之 后 数据 会 全 部 丢失 ,但 它 的 表现 跟 一 般 的 RDBMS 很 像 。( 其 实数 据 是 可 以 保存 下 
来 的 ， 请 访问 HSQLDB 的 网 站 了 解 更 多 细节 。) 

虽然 我 们 可 能 又 扔 给 你 两 项 新 技术 , 但 随 书 源码 中 的 构建 脚本 会 帮 你 把 正确 的 JAR 依 赖 项 
和 配置 文件 放 到 正确 的 地 方 。 


1 于 
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首先 ， EN 如 代码 清单 11-15 所 示 。 
代码 清单 11-15 ”用 于 HSQLDBt 


<?xMm] version="1 .0" encoding= "UITIF=9"?> 

<!IDOCTYPE hibernate-conf iguration PUBLIC 

"-//Hibernate/Hibernate Confiaguration DID 3.0//EN" 
"http://hibernate.sgourceforge .net/hibernate-configuration-3.0.dtd"> 


<hibernate-configuration> 
<SesSslion-=-factory> 


Droperty name="hibernate.dialect"s 设置 方言 
org.hibernate.dialect .SOLDIalLect Ws 


</property> 

<property name="hibernate.connection.driver class"> 
org.hsgaldb .jadbcDriver 
-</property> 


<Droperty name="hibernate.connection.url"> | TC 
jdbc:hsgldb:mem:wgjad EA 指定 要 连接 的 URL 
</property> 


<Droperty name="hibernate.connection.username">sa</property> 
<property name="hibernate.connection.password"s</propertys> 
<property name="hibernate.connection.autocommit">true</property> 
<Droperty name="hibernate.hbm2addl .auto"> 


Create | i , 
</property> | 自动 创建 数据 表 
<Property name="hibernate.show sql"strue</property> 


<mapping resource="Ticket .hbm.xml"/s i 
</session-factory> " 
i 0 映射 Ticket 类 


</hibernate-configurations 

你 应 该 注意 到 了 , 清单 中 的 最 后 一 行 语句 引 用 了 Ticket 类 的 映射 资源 ( <mapping resource=" 
Ticket ,hbm.xml"/> ) 大。 这 个 资源 会 告诉 Hibemate 怎 么 把 Java 文 件 映 射 到 数据 库 列 。 在 Hibernate 
配置 文件 里 ,除了 方言 ( HSQLDB )， 还 有 所 有 Hibemate 需 要 用 来 在 幕后 自动 构建 SQL 的 信息 。 

人 尽管 Hiberate 允 许 你 在 Java 类 里 直接 用 注解 添加 映射 信息 , 但 我 们 还 是 更 喜欢 下 面 这 种 XML 
映射 方式 ， 如 代码 清单 11-16 所 示 。 


警告 ”注解 跟 XML 映 射 之 间 的 选择 之 战 在 邮件 列表 中 已 经 打 了 很 久 了 ， 所 以 你 最 好 选 个 自己 喜 
欢 的 ， 然 后 就 由 它 去 吧 。 


代码 清单 11-16 用 于 Ticket 的 Hibernate 虹 
到 了 其 人 versicn="1 .0" encoding="UTF-8"?> 
<IDOCTYPE hibernate-mapping PUBLIC 
"-//Hibernate/Hibernate Mapping DTD 3.0//EN" 
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd"s> 


<hibernate-mappings 


“Class | 
name= "com.JavaTdeveloper .chapter11 标 出 要 映射 的 类 
“® ,liasting 11 18.Ticket"> 
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i name="ticketId"™ 


type="long" 指定 ticketI4 为 关键 字 
column="ID" /> 
broperty name="faceVvalue" 
type="java.math.BigDecimal' facevalue 映 射 


column="FACE VALUE" 
not-null=e"ftalse" /> | 


<property name="discountRate” 


type="jJava.math.BigDecimal" i 
colum="DISCOUNT RATE" 1iscountRate 映 身 


not-null="true" /> 


</Class> 
</hibernate-mapping> 


弄 完 配置 文件 , 该 想 想 测 什 么 了 。 用 唯一 ID 获取 micket 是 业务 需要 。 为 了 满足 这 一 业务 ( 和 
Hibernate 映 射 ) 要求， 必须 将 ricket 类 改 成 代码 清单 11-17 这 样 。 
代码 清单 11-17 带 有 ID 的 ricket 让 FY : 


import java.math.BigDecimal; 


有 Tp 和 , 和 
1 i sal 4 i ' = - ‘ 
i 1 


public class Ticket { 
Public static final int BASIC TICKET PRICE = 30; 


private long ticketId,; | 
private final Price priceSource; | 加 上 ID 


private BigDecimal faceValue = nmull; 
private BigDecimal discountRate; 


private final class FixedPrice implements Price | 
public BigDecimal getInitialPrice() | 
return new BigDecimal (BASIC TICKET PRICBE) ; 
} 
} 


public Ticket (long id) 1 
tickaetId = id; 
priceSource = new FixedPprice|): 
discountRate = new BigDecimall(l"l.0").; 


| 


public void SetcTicketId(1ong ticketId) 1 
this.ticketId = ticketlId; 


} 


public long getTicketId{) { 
return ticketId:; 

} 

public void setFaceValue (BigDecimal faceVvalue) |{ 
this.faceValue = faceValue, 


| 


public BigDecimal getFaceValua() | 
return faceValue: 
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| 


publiec void setDiacountRate (BigDecimal discountRate) | 
this.discountRate = discountRate; 


} 


public BigDecimal getDiacountRate() | 
_ return discountRate:; 
} 
public BigDecimal getDiscountprice() | 
if (faceValue == Null) faceValue = priceSource.getinitialPerice(); 
return faceValue.multiply ldiscountRate).,; 
| 
| 
现在 Ticket 的 映射 有 了 ，micket 类 也 改过 了 ， 可 以 调用 micketHibernateDao 里 的 
findTicketByI6 方 法 进行 测试 了 。 哦 , 还 要 写 JUnit 测 试 设置 的 准备 代码 , 如 代码 清单 11-18 所 示 : 


代码 清单 11-18 TicketHibernateDaoTest 测 试 类 
import Java.math.BigDecimal; 
import org.hibernate .cfg.configuration; 
import org.hibernate .SessionFactory; 
import org.jJjunit.*, 
import static org.junit .Assert.*:; 


public class TicketHibernateDaoTest 1 


private static SessionFactory factory:; 
private static TicketHibernateDao ticketDao; 
private Ticket ticket; 

private Ticket ticket2; 


theforeClasas 
public static void baseSetUp() | 

factory = 

new Configuration(). 
configure() .buildSessionFactory(); 使 用 Hibernate 配 置 

ticketDao = new TicketHibernateDao lfactory); 
} 
Before 
public void setUpTest () 
| 


Eicket = new Ticket (1); 


ticketDao.save (ticket):; 设置 测试 mieckeat 
ticket2 = new Ticket (2); 的 数据 
ticketDao.,save (ticket2); 

} 

TEBt 

Public void findTricketByIdHappyPath() throws Exception I 
Ticket ticket = ticketDao.findTricketById(1); 
assertEquals (new BigDecimal ("30.0"), © 找到 Ticket 

ticket .getDiscountPrice())}); 可 一 
} 
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After 

public static void tearDown{) 1 
ticketDao.delete (ticket}):; |; i 
ticketDao.delete (ticket2).; 清除 数据 

} 

AfLercCla 

public static void baseTearDown() | ] 关闭 
factory.closel():; 

| 


在 运行 任何 测试 之 前 ， 先 用 Hibernate 的 配置 创建 所 要 测试 的 DAO 和 .然后 , 在 每 个 测试 运行 

之 前 ， 都 在 HSQLDB 数 据 库 里 存 两 条 门票 的 记录 ( 作为 测试 数据 ) @@。 运 行 测试 ,测试 DAO 的 
findTicketById 方 法 估 。 

因为 你 还 没 写 TicketHibernateDao 类 及 其 方法 ， 所 以 测试 一 开始 会 失败 。 使 用 Hibernate 

框架 不 需要 SQL, 也 不 需要 提 及 用 的 是 HSQLDB 数 据 库 。 因此 , DAO 的 实现 应 该 和 代码 清单 11-19 


代码 清单 11-19 TicketH: 


import Java.util.Ligst:; 

import org.hibernate.Criteria; 

import org.hibernate.Sesgion; 

import org.hibernate.SessionFactory,; 

import org.hibernate.criterion.Restrictions.; 


public class TicketHibernateDao | 


private static SessionFactory factory; 
private static Seession Session: 


public TicketHibernateDao lSesgsionFactory factory) 
| 
TicketHibernateDao.factory = factory:; ei 
TicketHibernateDao.sesgsion = getSessicon'(),; ] 设置 工厂 和 会 话 
} 
Public void Save (Ticket ticket) 
| 
egslion .Save (ticket).: ] 了 保存 Ticket 
session.flusht():; : 
} 
Publie Ticket findTicketById (long ticketId) 
| 
Criteria criteria = 
Session.createCriteria (Ticket .clagas): 
criteria.addlRestrictions.eg("ticketIid", ticket1Id})}: 
List<Ticket>s tickets = criteria.list(}:; 
return tickets.get (0}); 


了 使 用 fID 查找 Ticket 
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public void delete(Ticket ticket) | 
sesgsion.delete (ticket)}),; 
SesBgsion. flushl(), 
| 
private static synchronized Session getSession() { 
Teturn factory.openSession{); 


}. 
} , 
DAO 的 save 方 法 特别 不 起 有 上 腿 ， 就 是 调用 Hibemate 的 save 方 法 ， 然 后 用 fl1ush 确 保 对 象 能 存 
到 HSQLDB 数 据 库 中 八 。 要 取出 Ticket， 可 以 用 Hibemate 的 criteria (相当 于 SQL 里 的 wHERE 
从 句 ) @。 
写 完 DAO 之 后 ， 测 试 就 能 通过 了 。 你 可 能 已 经 注意 到 了 ，save 方 法 也 已 经 被 部 分 测试 到 了 。 
你 可 以 继续 写 更 加 完整 的 测试 ， 比 如 检查 一 下 从 数据 库 中 取 回 的 票 是 否 带 有 正确 的 discount- 
Rate。 现 在 可 以 提前 测试 数据 库 访 问 代 码 了 ,所 以 数据 库 访 问 层 也 得 到 了 TDD 方 式 的 所 有 好 处 。 
我 们 接着 讨论 下 一 个 测试 蔡 身 : 模拟 对 象 。 


11.2.4 ”模拟 对 象 


模拟 对 象 跟前 面 提 过 的 存根 对 象 是 亲戚, 但 存根 对 象 一 般 都 特别 果 。 比 如 在 调用 存根 时 它们 
通常 总 是 返回 相同 的 结果 。 所 以 不 能 模拟 任何 与 状态 相关 的 行为 。 

看 个 例子 : 假设 你 想 用 TDD 方 式 写 一 个 文本 分 析 系 统 。 其 中 一 个 单元 测试 要 求 文本 分 析 类 对 
某 第 博文 中 出 现 的 “Java 7” 进 行 计数 。 但 这 篇 博文 是 第 三 方 资源 ， 所 以 很 多 失败 都 跟 你 写 的 计 
数 算法 没 太 大 关系 。 换 句 话 说， 测试 代码 不 是 孤立 的 , 并且 获取 第 三 方 资源 可 能 很 费时 间 。 下 面 
是 一 些 很 常见 的 失败 : 

口 由 于 防火 墙 限制 ， 你 的 代码 可 能 无 法 访问 互联 网 上 的 这 篇 博文 ; 

口 这 篇 博文 可 能 被 挪 走 了 ， 而 链接 没有 重 定向 ; 

口 博文 可 能 被 编辑 过 ,“Java 7” 出 现 的 次 数 可 能 增加 了 ， 也 可 能 减少 了 。 

用 存根 几乎 不 可 能 把 这 个 测试 写 出 来 ， 即 便 能 写 也 极其 繁 珊 , 模拟 对 象 此 时 登场 。 这 是 一 种 
特殊 的 测试 欧 向 ， 你 可 以 把 它 当 做 可 以 预 编 程 的 存根 或 超级 存根 。 使 用 模拟 对 象 非常 简单 : 在 准 
备 要 用 的 模拟 对 象 时 ,告诉 它 预计 会 有 哪些 调用 ,以 及 每 个 调用 该 如 何 响应 。 模拟 会 跟 DI 结 合 得 
很 好 ， 你 可 以 用 它 注 入 一 个 虚拟 的 对 象 ， 这 个 对 象 将 完全 按照 已 知 方式 行动 。 

让 我 们 看 一 个 剧院 门票 的 例子 。 我 们 会 用 一 个 流行 的 模拟 类 库 Mockito ( http://mockito.org/ )， 
请 看 代码 清单 11-20。 


import static org.mockito.Mockito.™; 
import static org. junit ,Assert.*; 


import java.math.BigDecimal:; 
import org.Junit .Test,; 


1 入 
Yr 


296 第 11 章 测试 驱动 开发 


public class TicketTeat | 创建 模拟 对 象 
二 Test : 
public void tenPercentDiscount () | 
Price price = mock (Price.class):; ! 


when (price.getIinitialPrice(}). 对 模拟 对 象 编程 
mm. thenReturnlnew BigDecimal ("10")); 以 便 进行 测试 


Ticket ticket = new Ticket (price, new BigDecimal ("0.9")}; 
assertEBquals(9.0, ticket .getDiscountPrice() .doubleVvalue{(}, 0.000001); 


verify (price}) .getIinitialPricel(), 
| 
| 


创建 模拟 对 象 需 要 调用 静态 的 mock() 方 法 人 @， 并 将 模拟 目标 类 型 的 class 对 象 作为 参数 传 给 
它 。 然 后 要 把 模拟 对 象 需要 表现 出 来 的 行为 记录 下 来 ， 通 过 调用 when{( ) 方 法 表明 要 记录 哪些 方 
法 的 行为 ， 然 后 用 thenReturn ( ) 指定 所 期 望 的 结果 是 什么 人 四 .最 后 要 证 实在 模拟 对 象 上 调用 了 
预期 的 方法 。 这 是 为 了 确保 你 的 正确 结果 不 是 经 由 不 正确 的 路 征 得 到 的 。 

你 可 以 像 使 用 常规 对 象 那样 使 用 模拟 对 象 , 并 且 无 需 任 何其 他 步骤 就 可 以 把 它 传 给 你 调用 的 
Ticket 构 造 方法 。 这 使 得 模拟 对 象 成 为 了 TDD 的 得 力 工 具 ， 有 些 从 业者 实际 上 更 喜欢 所 有 事情 
都 用 模拟 对 象 来 做 ， 完 全 放弃 了 其 他 测试 替身 。 

不 管 你 是 不 是 选择 这 种 “最 模拟 ”的 TDD 风 格 ， 完 整 的 测试 替身 ( 需要 的 话 加 上 一 点 DI) 知 
识 会 让 你 训 不 自 惧 地 进行 重 构 和 编码 ， 即 便 面 对 复杂 的 依赖 和 第 三 方 子 系统 也 不 怕 。 

Java 开 发 人 员 会 发 现 TDD 的 工作 方式 非常 容易 上 手 。 但 Java 经 党 伴随 着 一 个 反复 出 现 的 问 
题 一 一 有 些 繁琐 。 在 纯粹 的 Java 项 目 中 用 TDD 会 导致 大 量 的 套路 化 代码 。 好 在 现在 你 已 经 学 了 一 
些 其 他 的 JVMi 语 言 ， 能 用 它们 做 出 更 精炼 的 TDD。 实 际 上 ， 从 测试 开始 将 非 Javai 语 言 带 入 项 目 中 
是 推动 凶 鸽 言 项 目的 经 典 方式 之 一 。 

在 下 一 扩 中， 我 们 会 讨论 ScalaTest， 这 个 测试 框架 有 具有 广泛 的 测试 用 途 。 我 们 会 从 介绍 
ScalaTest 开 始 ， 并 会 向 你 展示 如 何 用 它 运 行 JUnit 测 试 来 测试 Java 类 。 


11.3 ScalaTest 


如 果 你 还 记得 ， 我 们 在 7.4 节 说 过 TDD 是 动态 语言 的 理想 用 例 。 实 际 上 ，Scala 先 进 的 类 型 推 
断 让 它 在 做 测试 上 同样 也 有 很 多 优势 ， 尽 管 它 是 静态 类 型 系统 ， 还 是 经 常会 让 人 觉得 它 是 动态 

Scala 中 的 主 测试 框架 是 ScalaTest。 它 为 做 各 种 测试 提供 了 一 些 极 其 实用 的 特质 和 类 一 一 从 
JUnit 风 格 的 单元 测试 到 全 面 的 集成 和 验收 测试 。 我 们 来 看 一 个 ScalaTest 的 实战 例子 。 

代码 清单 11-21 用 SecalaTest 重 写 了 11.4 节 中 的 代码 ， 并 且 加 了 一 个 新 的 sellmicket1() 方 法 测 


试 fiftyDiscountTickets()。 
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代码 清单 11-21 ScalaTest 凡 
import java.math.BigDecimal 
import java.lang.lllegalArgument Exception 
import org.scalatest .Junit .JUnitSuite 
import org.scalateast .junit .ShouldMatchersForJUnit 
import org.Junit .Test 
import org.junit .Before 
import org. junit.Assert,. 


class RevenueTest extends JUnitSuite with ShouldMatchersForJUnit I 
var venueRevenue: TicketRevenue = 


Before def initializel() | 
venueRevenue = new TicketRevenuel|) 
} 
aTest def zeroSalesBqualsZeroRevenue|() | 
assertEquals (BigDecimal .ZERO, venueRevenue estimateTotalRevenue 0}):; 
| 
aTest def failIlfTooManyOrTooFewTicketsAreSold() 1 
evaluating (| venueRevenue .estimateTotalRevenue|-1) |) ] 预期 的 昼 常 
ms Should produce [IllegalArgument Exception)] 一 一 
evaluating {| venueRevenue .estimateTotalRevenue (101) | 
mh Bhould produce [IllegalArgument Except1ion)] 


| 


ETeest def tenTicketasScoldIeThreeHundreadInRevenue (|) 1 
Val expected = new BigDecimal ("300"); 
assert (expected == VenueRevenue .estimateTotalRevenue(10)).; 
} 
STest def fiftyDiscountTickets() 1 
for (i «<-= 1 te 50] 
a VenueRevenue.sellTicket (new Ticket|(}) 
for (1 <- 1 to 50) 
nm VenuUeRevenue. Sel lTicket (new Ticket (new StubPrice'(), 
sp new BigDecimal (0.9))) 


| et | | : Re ] Scala 风 格 的 断言 
= VenueRevenue.getRevenue(}) .doubleVvalue (})});} 


| 
| 


我 们 还 没 讲 过 Scala 如 何 处 理 注解 。 它 们 看 起 来 跟 Java 注 解 一 样 。 这 没什么 好 说 的 。 你 的 调试 
也 是 放 在 扩展 了 aunitsuit 的 类 中 ， 这 就 是 说 ScalaTest 会 把 这 个 类 当做 它 能 运行 的 东西 。 

你 可 以 在 命令 行 中 用 本 地 ScalaTest 运 行 器 轻松 运行 ScalaTest: 

ariel:scalatest boxcats scala -cp /Users/boxcat/projects/tickets.jar: /Users/ 
boxcat /projects/wgjd/code/lib/,acalatest-1.6.1.jar:/Users/boxcat/ 


projects/wgjd/code/lib/junit-4.8.2.|jar org.scalatest .toola.Runner -© -8 
com. javaTdeveloper .chapterll .scalatest .RevenueTegst 


在 这 条 命令 中 ， 所 测试 的 Java 类 放 在 tickets.jar 文 件 中 ， 所 以 要 把 它 跟 ScalaTest 和 JUnit 文 件 一 
起 放 在 类 路 径 中 。 
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这 条 命令 用 -s 选 项 指定 了 要 运行 的 测试 集 ( 省 略 -s 选 项 会 运行 所 有 测试 集中 的 所 有 测试 ), -o 
选项 把 测试 输出 发 送 到 标准 输出 中 ( 用 -e 把 测试 结果 输出 到 标准 错误 流 中 )。ScalaTest 参 照 这 个 
配置 输出 报道 途径 ( 包括 其 他 途径 ， 比 如 图 形 化 界面 )。 前 面 的 例子 产生 的 输出 如 下 所 示 : 

Run eatarting. Expected test count is: 4 

RevenueTest : 

=- ZEroSalesEgqualsZeroRevenue 

- failIfTooManyOrTooFewTicketsAreSold 

-~ tenTicketsSoldIsThreeHundredInRevenue 

- fifttyDiscountTickets 

Run completed in 820 milliseconds. 


Total number of tests run: 4 

Suites: completed 1, aborted 0 

Tests: succeeded 4, failed 0, ignored 0, pending 0 
All tests passed. 


这 些 测试 已 经 被 编译 进 了 一 个 类 文件 中 。 只 要 类 蹄 径 中 有 JUnit 和 ScalaTest 两 者 的 JAR， 就 可 
以 用 scala 运 行 这 些 测 试 ， 市 不 用 在 JUnit 运 行 闫 中 。 

ariel:scalatest boxcat$ scala -cp /Users/boxcat /projects/tickets.Jar:/Usergl/ 
boxcat /projecta/wgijd/code/lib/acalatest-1.6.1.1ar:/Usera/boxcat/ 
projects/wgjd/code/lib/junit-4.8.2.jar org.junit .runner.JmnitCore 
com. JavaTrdeveloper.chapterll.scalateast .RevenueTeat 

JUnit version 4.8.2 

i 0 096 

OK (4 tests) 

当然 ， 输 出 会 稍 有 不 同 ， 因 为 执行 测试 用 的 是 不 同 的 工具 (JUnit 运行 旭 )。 


注意 ”在 用 Maven 构 建 秆 12 章 的 java7developer 项 目 时 ， 我 们 会 用 这 个 JUnit 运 行 器 。 


为 项 目 中 的 主要 编程 


我 们 在 这 一 节 主要 讨论 用 ScalaTest 测 试 


Scala 是 稳定 层 语言 ， 所 以 如 果 你 在 使 用 Scala 代 码 ， 应 该 也 可 以 像 测试 Java 
la 代码 库 。 所 以 用 ScalaTest 代 替 JUnit 是 使 用 TDD 方 式 的 不 二 之 选 。 


快速 了 解 ScalaTest 后 ， 我们 对 TDD 的 讨论 就 结束 了 。 第 14 章 对 行为 驱动 开发 ( BDD ) 的 讨论 
就 是 建立 在 这 些 内 容 之 上 的 ， 从 逻辑 关系 上 可 以 将 BDD 看 成 TDD 的 下 一 步 。 


11.4 小结 
测试 驱动 开发 能 消除 或 减轻 开发 过 程 中 的 臣 惧 。 遵 从 TDD 风 格 ， 比 如 单元 测试 的 红 一 绿 一 重 
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构 循环 ， 开 发 人 员 可 以 把 自己 从 思维 定式 中 解放 出 来 ， 不 会 步 人 临时 拼 岩 代码 的 窗 境 。 

JUnit 是 Java 开 发 人 员 的 主要 测试 类 库 。 它 可 以 指定 设置 和 拆 印 挂 物 ， 运行 一 个 测试 集 里 相互 
独立 的 测试 。JUnit 的 断言 机 制 会 判断 调用 实现 逻辑 后 是 否 能 产生 想 要 的 结果 。 

不 同类 型 的 测试 替身 可 以 帮 你 写 出 恰当 的 测试 。 你 可 以 用 四 种 测试 替身 ( 虚设 、 存 根 、 伪 装 
和 模拟 ) 取代 依赖 项 ， 从 而 让 测试 精准 运行 。 在 编写 测试 代码 时 ,借助 模拟 对 象 可 以 实现 终极 的 
灵活 性 。 

ScalaTest 始 终 秉持 大 量 减少 套路 化 测试 代码 的 观念 有 助 于 开发 人 员 深信 理解 测试 的 行为 驱 
动 开发 风格 。 

我 们 在 下 一 章 讨论 自动 构建 ， 以 及 建立 在 TDD 基 础 之 上 的 持续 集成 (CI ) 开发 方法 。 使 用 
CI 开发 方法 ， 你 能 立即 得 到 每 个 新 变化 的 自动 反馈 ， 并 且 它 鼓励 开发 团队 成 员 之 间 彻 底 透 明 化。 
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本 章 内 容 

口 构建 管道 和 持续 集成 (CI ) 的 重要 性 

口 Maven 3: 惯例 优先 于 配置 的 构建 工具 
DJenkins: 得 到 公认 的 CI 工具 

口 使 用 FindBugs 和 Checkstyle 等 静态 代码 分 析 工 具 

口 Leiningen: Clojure 构 建 工 具 


我 们 接 下 来 要 讲 的 故事 取材 于 MegaCorp 的 真实 事件 ,出 于 对 当事人 的 保护 隐 去 了 真实 姓名 。 
故事 的 主角 是 

口 Riley， 刚 毕业 的 新 人 ; 

口 Alice 和 Bob， 两 个 “经 验 丰富 ”的 开发 老手 ; 

口 Hazel， 紧 张 的 项 目 经 理 。 

时 间 是 周 五 下 午 两 点 ， 新 开发 的 Sally 支 付 功能 要 在 周末 跑 批 前 上 线 。 

Riley: 我 能 为 上 线 做 点 什么 吗 ? 

Alice: 当然 ， 我 想 最 后 一 版 是 Bob 构 建 的 。Bob? 

Bob: 是 的 ， 是 我 几 周 前 用 Eclipse 生成 的 。 

Riley: 但 现在 我 们 都 用 IntelliJ 了 ; 那么 ,该 怎么 构建 呢 ? 

Bob: 哦 ， 这 需要 些 经 验 ! 总 之 我 们 会 搞定 它 ， 年 轻 人 ! 

Riley: 好 。 我 设 这 方面 的 经 验 ， 但 支付 功能 的 构建 应 该 没 问题 ， 对 吧 ? 

Alice: 当然 没 问题 。 我 在 两 周 前 刚 创 建 的 代码 分 支 ， 其 他 人 对 代码 的 改动 肯定 还 不 多 。 

Bob: 但 是 ， 实 际 上 ， 你 知道 我 们 添 了 些 泛 型 的 修改 ， 对 不 对 ? 

[ 槛 检 的 沉 睦 ] 

Hazel; 改 完 你 们 要 经 常 在 一 起 试 坛 。 这 个 我 们 强调 过 很 多 次 了 1 

Riley: 要 不 要 我 订 外 卖 ? 和 貌似 今 晚 我 们 得 加 班 了 。 

Hazel: 你 说 对 了 ， 学 得 挺 快 嘛 ! 

Alice: 实际 上 ， 我 已 经 将 订餐 电话 设 成 快速 拨号 状态 了 ， 这 是 常态 1 
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Hazel: 赶紧 把 它 搞定 ! 我 们 已 经 因为 延迟 发 布 和 bug 太 多 损失 很 多 了 ， 高 管 正 想 找 机 会 杀 鸡 
做 猴 呢 。 

Alice、Bob 和 Riley 明 显 没有 优秀 的 构建 和 持续 集成 (CI ) 经 验 , 但 “构建 和 CI” 究 竟 是 什么 
意思 ? 

构建 和 持续 集成 ”快速 和 重复 地 为 各 种 环境 产生 高 质量 二 进 制 部 署 工 件 的 过 程 。 

开发 团队 经 常 谈论 “构建 ”或 “构建 过 程 ””。 就 本 章 而 言 ， 我 们 在 提 到 构建 时 是 指 遵 循 构 
建 周期 用 构建 工具 将 源码 转化 成 二 进 制 工件 的 过 程 。 像 Maven 这 种 构建 工具 有 很 长 的 、 详 细 的 构 
建 周期 ， 它 们 中 的 大 多 数 对 于 开发 人 员 来 说 是 不 可 见 的 。 一 个 相当 基础 的 、 典 型 的 构建 周期 如 图 
12-1 所 示 。 


清除 一 > 编译 一 测试 一 > 打包 
图 12-1 一 个 简化 的 典型 构建 周期 


持续 集成 是 指 团 队 成 员 按照 “尽早 提交 ， 经 常 提 交 ” 的 口号 频繁 地 集成 工作 成 果 。 每 个 开发 
人 员 至 少 按 天 把 代码 提交 到 版 本 控制 系统 中 ，CI 服 务 器 会 自动 定期 构建 ,以 尽快 检查 集成 错误 2 。 
Cl 服务 器 通常 会 在 大 屏幕 上 显示 开心 /悲伤 的 表情 给 团队 以 反馈 。 
那么 构建 和 CI 为 什么 重要 ? 本 章 的 每 一 节 都 会 强调 某 些 独特 的 好 处 , 表 12-1 中 列 出 了 其 中 最 
为 重要 的 几 个 。 
表 12-1 构建 和 CI 的 主要 优势 
主 题 六 解释 
重复 性 任何 人 都 可 以 随时 随地 运行 构建 。 也 就 是 说 整个 开发 团队 都 可 以 自如 地 运行 构建 , 而 不 
需要 一 个 专门 的 “构建 负责 人 ”做 这 件 事 。 如 果 一 个 新 加 入 的 团队 成 员 需 要 在 周 日 的 凌 
晨 三 点 运行 构建 ， 他 可 以 毫 不 犹 驹 地 这 么 干 
尽早 反馈 -日 出 了 问题 ， 你 马上 就 能 知道 。 在 开发 者 处 理 需 要 集成 的 代码 时 这 跟 CI 尤 其 相关 
一 致 性 你 知道 部 署 的 软件 是 什么 版 本 ， 并 且 完 全 清楚 每 个 版 本 的 代码 
依 粮 管理 大 名 数 Java 项 目 都 有 几 个 依 球 项 ， 比 如 log4j)、Hibermate 、Guice 等 。 手工 管理 这 些 依 档 项 
可 能 会 非常 困难 , 而 且 有 一 个 版 本 发 生变 化 就 可 能 会 守 至 软件 不 可 用 , 良好 的 构建 和 CI 
能 确保 你 总 是 针对 同一 个 第 三 方 依赖 项 进行 编译 和 运行 


为 了 将 源码 部 署 到 运行 时 环境 中 ， 需 要 经 过 构建 周期 将 其 转变 成 二 进 制 工件 (JAR、WAR、 12 
RAR、EAR 等 )。 比 较 老 的 Java 项 目 通常 都 使 用 Ant， 而 比较 新 的 则 使 用 Maven 或 Gradle。 很 多 开 
发 团队 还 有 夜间 集成 构建 ， 有 些 已 经 升级 成 用 CI 服务 器 定期 执行 构建 了 。” 


山 如 果 你 的 团队 在 谈论 这 些 内 容 时 或 虐 诚 、 或 害怕 ， 或 话 不 多 ， 那 这 一 章 就 是 为 你 准备 的 。 
四 构建 时 间 可 配置 间 珊 可 以 是 几 分 钟 ， 也 可 以 在 提交 时 触发 ， 或 在 其 他 特定 时 间 运 行 。 
加 合作 极 其 默契 的 项 目 团队 能 让 非 技术 队友 运行 构建 。 
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警告 如 果 你 从 IDE 中 构建 JAR 文 件 或 其 他 工件 ， 那 是 在 自 找 麻烦 。 从 IDE 中 构建 得 到 的 不 是 与 
本 地 IDE 设 置 无 关 的 可 重用 构建 ， 那 简直 就 是 埋 下 了 祸根 。 作 为 朋友 ， 我 再 怎么 强调 这 人 一 
点 都 不 为 过 : 不 允许 你 用 IDE 构 建 工件 ! 


但 大 多 数 开 发 人 员 都 觉得 构建 和 CI 不 值得 他 们 投入 精力 ， 他 们 觉得 这 个 工作 做 起 来 不 够 爽 ， 
也 得 不 到 什么 回报 ,构建 工具 和 CI 服务 器 经 常 是 在 项 目 一 开始 的 时 候 措 起 来 ,但 很 快 就 被 遗忘 了 。 
这 么 多 年 来 ， 我 们 听 到 过 很 多 类 似 的 说 法 ;“ 我 们 为 什么 还 要 在 构建 和 CI 服务 器 上 花 时 间 呢 ? 现 
在 弄 得 也 挺 好 用 的 。 人 够 用 就 好 ， 对 不 对 ? ” 

我 们 坚信 息 好 的 构建 和 CI 能 加 快 编码 速度 ， 提 高 代码 质量 。 跟 TDD ( 见 第 11 章 ) 相 结 合 的 构 
建 和 CT 意味 着 你 可 以 毫 无 后 顾 之 忧 地 快速 重 构 。 你 可 以 把 它 当做 在 你 身后 默默 提供 支持 的 导师 ， 
它 为 你 营造 一 个 安全 的 环境 ， 让 你 可 以 快速 编写 并 大 胆 修改 代码 。 

本 章 ， 我 们 会 首先 介绍 Maven 3。Maven 3 是 一 个 流行 (还 有 争议 ， 有 些 开 发 人 员 挺 讨厌 它 ) 
的 构建 工具 , 会 强迫 你 按照 严格 定义 好 的 构建 周期 工作 。 介绍 Maven 3 的 内 容 中 , 除了 常见 的 Java 
代码 ， 还 会 涉及 Groovy 和 Secala 代 码 的 构建 。 

Jenkins 是 Cl 界 的 流行 天 王 ， 可 以 通过 多 种 方式 配置 ( 以 插件 系统 的 方式 ) 持续 执行 构建 ,并 
产生 质量 指标 。 在 学 习 Jenkins 时 ， 我 们 还 会 深入 了 和解 FindBugs 和 Checkstyle 产 生 的 代码 质量 指标 。 

在 学 完 Maven 和 Jenkins 之 后 ， 你 应 该 会 彻底 熟悉 典型 的 Java 构 建 和 CI 流程 。 之 后 我 们 会 重点 
讨论 Clojure 的 构建 工具 Leiningen ,完全 从 另 一 个 角度 看 看 构建 和 部 署 工 具 。 你 会 看 到 它 如 何在 提 
供 工 业 级 强度 的 构建 和 部 署 能 力 的 同时 实现 极其 迅速 、 易 用 的 TDD 风 格 。 

与 Maven 3 的 相遇 将 开局 你 的 构建 和 CI 之 旅 ! 


12.1 与 Maven 3 相册 


Maven 是 流行 的 Java 及 JVM 语 言 相关 的 构建 工具 ， 然 而 反对 它 的 人 和 支持 它 的 人 态度 同样 坚 
决 。 它 的 设计 理念 是 ， 严 格 的 构建 周期 辅 以 强大 的 依赖 管理 是 成 功 构建 的 必要 条 件 。Maven 不 仅 
是 构建 工具 , 更 是 项 目 技术 组 件 的 管理 工具 。 实际 上 , Maven 的 构建 脚本 叫做 POM ( Project Object 
Model， 项 目 对 象 模型 ) 文件 。 这 些 POM 文 件 是 用 XML 写 的 ， 并 且 每 个 Maven 项 目 或 模块 都 有 一 
个 pom.xml 文 件 。 


注意 POM 文 件 中 马上 要 加 入 对 备 选 语言 的 支持 ， 从 而 满足 用 户 对 灵活 性 的 要 求 ( 就 像 Gradle 
提供 的 那些 功能 )。 


PerryerynmyrryaPo 特别 pp 的 jan 二 pp 和 和 作 ppp rr 
的 一 段 时 间 。 我 们 不 准备 在 这 里 再 讲 了 ， 因 为 之 前 已 经 有 人 讲 过 上 百 次 了 。 更 关键 的 是 ,我 们 
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觉得 Ant 没 有 强制 实行 通用 的 构建 周期 ， 也 没有 一 组 通用 ( 强制 的 ) 构建 目标 。 这 就 是 说 开发 
人 员 必 须 研究 手头 每 个 Ant 构 建 的 细节 。 如 果 你 要 用 Ant，Ant 网 站 Spa 列 出 
了 所 有 必需 的 细节 。 

Gradle 是 这 一 领域 的 新 秀 。 它 有 意 选择 了 和 Maven 相 反 的 路 线 ， 限制 不 会 那么 严格 ， 你 可 
以 按 自己 的 方式 声明 构建 过 程 。 它 也 四 Maven 一 样 提供 依赖 管理 和 很 多 其 他 特性 。 如 果 你 想 举 
试 下 Gradle， 可 以 访问 Gradle 网 站 ( www.gradle.org ) 了 解 更 多 细节 。 

要 学 习 优 秀 的 构建 实践 ，Maven 是 适合 的 工具 。 它 强制 你 遵循 Mave 
这 个 构建 周期 ， 你 就 可 以 轻松 地 构建 世界 上 任何 一 个 Maven 项 目 。 


构 建 周 期 一 旦 掌握 


Maven 有 来 取 了 惯例 优先 配置 的 策略 ， 并 希望 你 能 融 人 到 它 的 世界 观 ， 在 源码 该 怎么 布局 、 属 
性 如 何 过 滤 等 设置 上 都 能 接受 它 的 安排 。 这 可 能 会 吓 着 基 些 开发 人 员 ， 但 Maven 的 构建 周期 是 经 
过 多 年 深思 部 上 谍 总 结 出 来 的 , 廊 看 它 提供 的 路 征 走 往往 是 最 合理 的 。 而 对 于 那些 极力 反对 墨 守 成 
规 的 人 ，Maven 确 实 提 供 了 和 杀 盖 默认 值 的 办 法 ,但 那样 会 做 出 更 加 繁 珊 , 并且 标 准 化 程度 更 低 的 
构建 脚本 。 

用 Maven 执 行 构 建 就 是 让 它 执行 一 个 或 几 个 目标 (代表 特定 任务 ， 比 如 编译 源码 、 运 行 测试 
圭 )。 目标 都 是 绑 到 默认 构建 周期 中 的 ， 如 果 你 要 求 Maven 执 行 测试 ( 如 mvn test )， 它 会 先 编 
伞 源 码 和 测试 代 公 。 向 言 之 ， 它 会 蝇 迫 你 章 守 正 确 的 构建 周期 。 

如 果 你 还 没 装 Maven 3， 请 参见 附录 A 中 的 A.2 节 。 在 完成 下 载 和 安装 之 后 ， 再 回 到 这 里 来 创 
建 你 的 第 一 个 Maven 项 目 。 


12.2 Maven 3 入 门 项 目 


Maven 遵 循 惯 例 优先 的 原则 ， 你 只 要 创建 一 个 快速 启动 项 目 ， 马 上 就 能 看 到 它 惯用 的 项 目 结 
构 。 它 喜欢 的 典型 项 目 结构 看 起 来 和 下 面 的 布局 类 似 。 
| -- pom-md 
| 二 main 
“= Tn 


com 
| -- Company 
| “ -一 project 
| App . java 
| “-- reaourceas 
= test 
= Java 
== Com 


- Company 
‘== proiject 
-=- AppTest .ava 
"== FeBoOuUrcees 
= target 
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依照 惯例 ，Maven 把 代码 分 成 了 main 和 test 两 部 分 。 它 还 创建 了 一 个 特别 的 resources 目 录 ， 
构建 工作 所 需 的 其 他 任何 文件 ( 比如 用 于 日 志 的 log4.xml 文 件 、Hibernate 配 置 文件 以 及 其 他 类 似 
资源 文件 ) 都 放 在 这 个 目录 下 。pom.xml 是 Maven 的 构建 脚本 ， 关 于 这 个 文件 的 详情 ， 请 参见 附 
录 E。 

如 果 你 是 多 语言 程序 员 ，Scala 利 Groovy 源码 跟 Java 源 码 的 结构 一 样 ， 只 是 Java 源 码 放 在 java 
目录 下 ， 而 它们 的 根 目录 分 别 是 scala 和 groovy。Java、Scala 和 Groovy 代 码 可 以 高 高 兴 兴 地 手 拉 手 
出 现在 同一 个 Maven 项 目 中 。 

target 目 录 是 构建 运行 后 才 会 创建 的 。 所 有 的 类 、 工 件 、 报 告 和 构建 产生 的 其 他 文件 都 会 出 
现在 这 个 目录 下 。 对 于 Maven 项 目 结构 的 完整 列表 ， 请 参见 Maven 网 站 上 的 Introduction to the 
Standard Directory Layout ( 标准 目录 布局 介绍 ) 页 和 面 (http://t.cm/aKJYxo )。 

要 为 新 项 目 创 建 这 个 结构 ， 请 执行 下 面 的 目标 (注意 其 中 的 参数 ); 

mn archetype:generate 

-Dgroupld=scom,. mycompany , app 
-DartifactId=my-app 


-DarchetypeArtifactIid=maven-archetype-gquickstart 
-DinteractiveMode=false 


然后 你 会 看 到 Maven 开 始 刷 屏 ， 它 在 疯狂 下 载 插件 和 第 三 方 类 库 。Maven 需 要 它们 来 运行 这 
个 目标 ， 它 的 默认 下 载 地 址 是 Maven Central ( 工件 的 在 线 资源 库 )。 


— | 时 -2 pe | 可 
[3 + | | I | | i | 
[对 ie en se a MPL Ls | pt ~’ 人 | I 
we i 5 。 1 Ea 重 
, 其 _ 


d f MM 
4 | “= 上 名 


… 噬 ， Ey ge yy 目 pp 得 这 真 
是 Maven 的 错 吗 ? 我 们 认为 它 这 样 做 有 两 个 根本 原因 。 一 是 第 三 方 类 库 开 发 人 员 对 包 和 依 不 的 
管理 很 烂 ( 比如 在 他 们 的 pom xml 文件 里 指定 一 个 实际 上 并 不 需要 的 依赖 项 )。 另 一 个 是 继承 
自 JAR 为 主 的 包 系 统 自身 的 缺陷 ， 没 办 法 做 更 细 化 的 依赖 项 控制 。 


除了 “正在 下 载 ……”， 控 制 台 应 该 还 会 有 下 面 这 种 声明 ; 

[INEO] -==---=---------------------------- 7----------------------------- 
[INFO] BUILD SUCCESS 

[INFO] -------------- --=--- --------------------------------------------- 
[INFO] Total time: 1.7038 

[INFO] Finished at: Fr Jun 24 13:51:59 BST 2011 

[INFO] Final eile EM/ 16M 

[INFO] -=---=---------- 


如 果 这 一 步 失 败 了 ， 很 可 能 是 你 的 代理 服务 器 不 允许 访问 Maven Central, 捅 件 和 第 三 方 类 库 
都 放 在 那 上 面 。 要 解决 这 个 问题 ， 只 要 编辑 settings.xml 文 件 ( 见 附录 A 的 A.2 节 )， 把 下 面 这 部 分 
内 容 加 上 去 ， 请 根据 你 的 实际 情况 为 各 元 素 填 上 恰当 的 值 : 


吕 模 因 ( Meme ) 也 称 为 米 姆 、 弥 、 弥 因 、 弥 母 、 迷 因 以 及 迹 米 等 ,是 文化 资讯 传承 单位 。 这 个 词 是 1976 年 理 查 
德 ， 道 金 斯 在 《上 月 私 的 基因 ， 一 书 中 创造 的 ， 以 生物 学 中 的 演化 规则 类 比 文 化 传承 的 过 程 。 模 因 包 含 其 广 ， 包括 
宗教 、 许 言 、 新 闻 、 知 识 、 观 念 、 习 惯 、 习 俗 ， 其 至 口号 、 庆 语 、 用 语 、 用 字 、 笑 话 。 一 一 详 者 注 
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Droxies> 
rEONY> 
<activestrue</actives 
<protocol></protocol> 
<USername>< /USername> 
<pageword>< /passeword> 
<host»></host> 
Port></port> 


/proxy> 
</proxies> 


重新 运行 上 面 的 目标 ， 这 次 应 该 能 看 到 my-app 项 目 出 现在 了 目录 中 。 
提示 ”如果 团 队 中 的 所 有 人 都 遇 到 了 这 个 癌 题 , 请 在 $M2 HOME/confjsettings.xml 中 加 上 代理 配置 。 


Maven 支 持 的 原型 (项目 布局 ) 几乎 是 无 限 的 。 如 果 要 生成 某 个 特定 类 型 的 项 目 ( 比如 JEE6 
的 项 目 )， 可 以 执行 mvn archetype:generate 目 标 ， 然 后 只 要 遵照 它 给 你 的 提示 就 行 了 。 

为 了 探索 Maven 的 更 多 细节 ， 我 们 来 看 一 个 源码 和 测试 代码 都 已 经 淮 备 好 的 项 目 ， 用 它 把 整 
个 构建 周期 走 一 遍 。 


12.3 用 Maven 3 构建 Java7developer 项 目 


还 记得 图 12-1 中 的 构建 周期 吗 ? Maven 的 构建 周期 跟 那 个 类 似 ， 你 马上 就 要 经 历 构建 周期 中 
的 每 个 阶段 了 了。 尽管 本 书 中 的 源码 不 是 一 个 应 用 程序 ， 我 们 还 是 会 把 它们 统一 放 到 一 个 叫做 
java7developer 项 目 中 。 

这 一 节 的 重点 是 ; 

口 探索 Maven POM 文 件 ( 即 构建 脚本 ) 的 基础 ; 

口 如 何 编 译 、 测 试 和 打包 代码 ( 包括 Scala 和 Groovy ); 

口 如 何 用 环境 配置 处 理 多 个 环境 ， 

口 如 何 生成 一 个 包含 各 种 报告 的 项 目 网 站 。 

首先 你 要 搞 明 白 定 义 java7developer 项 目的 pom.xml 文 件 。 


12.3.1 POM 


java7developer 项 目 用 pom.xml 表 示 ， 包 括 各 种 插件 、 资 源 ， 以 及 构建 所 需 的 其 他 元 素 。 可 以 
在 解压 或 签 出 本 书 项 目 代 码 的 根 目录 (从 现在 开始 我 们 用 $BOOK_CODE 指 代 这 个 位 置 ) 中 找到 12 
这 个 pom.xml 文 件 。POM 主 要 由 四 部 分 组 成 ; 

D 项 目 基本 信息 ; 

O 构建 配置 ; 

吕 环境 配置 。 
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这 是 个 相当 长 的 文件 , 但 实际 上 它 没 有 看 起 来 那么 复杂 。 如 果 你 想 了 解 POM 中 可 以 包含 哪些 
内 容 的 完整 细节 ， 请 参见 Maven 网 站 上 的 POM Reference ( http://maven.apache.org/pom.html )。 
接 下 来 我 们 就 要 解释 java7developer 项 目 pom.xml 文 件 的 这 四 部 分 ， 先 从 项 目 基本 信息 开始 。 
1. 项 目 基本 信息 
ER 以 放 入 一 系列 的 基本 项 目 信 息 。 代码 清单 12-1 列 出 的 是 最 起 码 的 起 步 信息 。 


代码 清单 12-1 


<project ni / /maven. ep me 0.0" 
xmlns:xsi="http:/ /Ww.w3.org/2001/XMLSchema-instance" 
xaBi:schemaLocation="http://mnaven.apache.org/POM/4.0.0 

http://maven.apache.org/maven-v4 0 0.xsd"> 


<modelVersion>4.0.0</modelVersion» 
<aroupIldscom. Java7developer</groupld> 
<artifactIdsjavaTdeveloper</artiftactId> 
packaging>jar</packaging> 
<Vergionsl.0.0</version> 
name>javaTdeveloper<,/name> 
<descript ions 

proiject source code for 七 he book! 
</descriptions 
<Url>http:/ /www. javaTdeveloper .com</Url> 


<Droperties> 
<Broject .build.sourceEncoding> 


UrF-8 | 9 平台 无 关 的 字符 编码 
</project .build.sourceEncoding> 


</properties> 


这 个 工件 在 Maven 资 源 库 中 的 唯一 标识 符 由 三 部 分 组 成 : 第 一 部 分 是 <groupTd> 的 值 
com. java7developer®@: 第 二 部 分 是 <artifactId> 的 值 java7Tdeveloper。 <packaging> 
的 值 jar 告 诉 Maven 你 要 构建 一 个 JAR 文 件 ( 这 里 可 能 出 现 的 值 有 war、ear、rar、sar 和 和 har )。 
唯一 标识 的 最 后 一 部 分 是 <=version> 的 值 1.0.0”， 表明 版 本 号 (执行 Maven 发 布 时 会 在 这 个 值 
后 面 加 上 SNaPSsHOT )。 

文件 中 还 指定 了 <projectName> 和 <url> ， 以 及 一 些 其 他 可 选 的 项 目 信 息 和 .。 
<sourceEncoding> 为 UTF-8， 这 样 可 以 确保 在 所 有 平台 上 的 构建 都 是 一 致 的 合 。 

总 的 来 说 ， 这 个 配置 会 指导 Maven 构 建 出 java7developer-1.0.0.jar 工 作 ， 并 把 它 存 在 Maven 资 
源 库 中 的 com/java7developer/1.0.0 目 录 下 。 


3 . i 时 Pe ee ee ee 人 i | ee 荐 7 
i i ee 二 E La | 0 让 加 于 到 下 和 


| 
I 


pp rp 玖 :部分 rp 次 半 pp L 本 号 ， 并 
依照 惯例 在 版 本 号 后 面 加 上 -SNRAPSHOT 表 示 这 是 一 个 临时 性 的 工件 。 比 如 说 ， 在 你 的 团队 为 


上 版 本 号 的 格式 遵循 MajorMinor'Trivial 风 格 ， 这 是 我 们 的 量 爱 ! 


只 自 在 读书 @3 
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即将 发 布 的 1.0.0 版 本 持续 构建 JAR 时 ，Maven 会 依照 惯例 将 版 本 号 设置 为 1 .0 .0-sNAPSHOT。 
这 样 ， 各 种 Maven 插 件 就 知道 这 还 不 是 生产 版 本 ， 从 而 可 以 正确 处 理 它 。 在 把 这 个 工件 发 布 到 
生产 环境 中 时 ， 要 发 布 为 1.0.0， 下 一 个 修订 bug 的 版 本 要 从 1.0.1-SNRPSHOT 开 始 。 

Maven 通 过 它 的 发 布 插件 把 这 些 都 自动 化 了 。 要 了 解 更 多 细节 ， 请 参见 发 布 插件 页 面 
( http:/maven.apache.org/plugins/maven-release-plugin/ )。 现 在 你 已 经 明白 项 目 基本 信息 部 分 是 
什么 样 的 了 ， 接 下 来 我 们 来 看 看 <build> 吧 。 


2. 构建 配置 

<build> 中 包 仿 执行 Maven 构 建 周期 目标 所 需 的 插件 ”及 相应 的 配置 。 在 大 多 数 项 目 中 , 这 部 
分 内 容 都 相当 少 ， 因 为 通常 用 默认 插件 的 默认 设置 就 够 了 了 。 

在 java7developer 项 目 中 ，<buila> 中 有 几 个 覆盖 了 上 默认 值 的 捅 件 ， 以 便 可 以 : 

口 构建 Java 7 代码 ; 

口 构建 Scala 和 Groovy 代 码 ; 

口 运行 Java 、Scala 利 Groovy 测 试 ，; 

口 提供 Checkstyle 和 FindBugs 代 码 指标 报告 。 

捅 件 是 以 JAR 为 主 的 工件 ( 主要 是 用 Java 写 的 )。 要 配置 构建 插件 ， 需 要 把 它 放 在 pom.xml 文 
件 的 <puild><plugins> 中 。 跟 所 有 Maven 工 件 一 样 ， 每 个 插件 都 有 了 唯一 标识 ， 所 以 需要 指定 
<groupId>、<artifactId> 和 <version> 信 息 。 对 插件 的 所 有 配置 都 放 在 <configuration> 
中 ,并 且 每 个 插件 的 具体 配置 元 素 是 不 同 的 。 比 如 编译 插件 的 配置 元 素 有 <source>、 <target> 
和 <showwarnings>， 这 是 编译 器 独 有 的 配置 信息 。 

代码 清单 12-2 列 出 的 是 java7developer 项 目的 构建 配置 部 分 ( 完整 的 代码 清单 及 相应 的 解释 在 
附录 E 中 )。 


代码 清单 12-2 POM: 构建 信息 


<builds 
<Dluginss 
<plugins 
<groupId>org.apache .maven.plugins</groupld> 
“artifactId>maven-compiler-plugin</artifactId> 1 所 用 插件 


VeErSsSliOnNn>2.3.2</Versions 
Configurations 


OUrcesl ,TT</SOUrces ba 编译 Java 7 代码 


<targetsl,. Tc/targets 


<ShowWarnings>struec</showWarningss 


<ShowDeprecation>true</showDeprecations 四 
他 编译 器 警告 
forkstrue</forks 


中 如 果 你 需要 对 构建 进行 配置 ， 可 以 访问 Maven 的 插件 页 而 查看 插件 的 完整 列表 ( hittp://maven.apache. 
org/plugins/index.html ). 
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<executable»$ {jdk.javac.fullpath}</executable> 一 一 
</confiaguration> 
</plugin> javac 的 路 径 


plugin> 
< 器 roupId>org .apache .maven.plugins</grouplds 
<artifactId>maven-surefire-plugin</artifactlIds 
<Versions2.9</Versions 
<COnNnfigurations 
<eXAClUuUdes> 
<&excGludes | 
com/javaTdeveloper/chapterll/ 
= 1]isting 11 2/TicketRevenueTest .java 
/excludes> 
<eEXxClUude> 
com/ Javardeveloper/chapterlll 
= 1]isting 11 7/TicketTeast . Java | 
/exclude>s | 


排除 的 测试 


</excludes> 
</ Configuration> 
</plugins 
</plugins> 
/builds 


为 Maven 3 默认 是 编译 Java 1.5 的 代码 ， 而 我 们 要 编译 Java 1.7@， 所 以 需要 指明 编译 器 插件 
(的 版 本 ) @， 

既然 你 打破 了 惯例 , 所 以 还 要 加 上 几 个 编译 警告 选项 个 。 接 下 来 要 指定 Java 7 安装 在 哪儿 @，。 
只 需要 把 sample <os> build.properties 文件 另存 为 build.properties ， 并 编辑 其 中 的 jdk.javaec. 
fullpath 属 性 ， 因 为 Maven 会 用 到 它 。 

Surefire 插 件 是 测试 用 的 在 配置 中 我 们 排除 了 几 个 失败 测试 估 ( 有 两 个 第 11 章 的 TDD 测 试 )。 

现在 构建 部 分 已 经 讲 完 了 ， 可 以 进入 POM 中 非常 重要 的 部 分 了 : 依赖 管理 。 

3. 依赖 管理 

大 多 数 Java 项 目的 依赖 项 列表 都 很 长 ，java7developer 项 目 也 不 例外 。Maven Central Repository 中 
有 各 种 各 样 的 第 三 方 类 库 ， 所 以 Maven 可 以 帮 你 管理 这 些 依 赖 项 。 最 和 草 要 的 是 ， 这 些 第 三 方 类 库 都 
有 它们 目 己 的 pom.xml 文 件 ， 会 声明 各 目的 依赖 项 ，Maven 可 以 据 此 找 出 任何 需要 下 载 的 其 他 类 库 。 

这 些 依赖 项 一 开始 主要 分 为 两 个 作用 域 ( compile 和 test ) “。 设置 作用 域 跟 把 JAR 文 件 放 
到 CLASSPATH 下 是 一 样 的 效果 。 代 码 清单 12-3 是 java7developer 项 目的 <dependencies> 部 分 。 
完整 的 代码 清单 及 相应 的 解释 在 附录 E 中 。 


代码 清单 12-3 ”POM: 依赖 


<dependencies> 


dependency> 
山 J2EEJEE 项 目 通 营 也 会 用 到 muntime 作 用 域 的 依赖 项 。 
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<artifactld>sguice</artifactIds> 
vergions3.0</versions 


<BCope>compile</Scope> 
</dependency> 
<dependency> , i 
<groupld>javax.inject</groupIds 加 编译 作用 域 


<artifactIdsjavax.inject</artifactIds 
<Vergion>l</versions 
<ScCope>scompile</Scopes 

</dependency> 


<groupId>com.google.inject«</groupId> 
D 工件 的 瞧 一 ID 


<dependency> 
<groupId>junit</grouplid> 
<artifactId>junit</artifactIds> 
<VerBion>=4.8.2</version> 


<BCoOpe>test</Scope> 可 一 
</dependency> 
-dependencys>s \ : 
<gqroupId>org .mockito</groupId> 测试 作用 域 


<artifactIidsmockito-all</artifactIds 
Vergionsl1 .068.5c/Versions 
<SCOPe>test</acope> 

</dependency> 


i 

为 了 让 Maven 找 到 你 引用 的 工件 ,需要 给 它 正 确 的 <groupId>、<artifactIdG> 和 <version>@，。 
我 们 在 之 前 提 到 过 ,把 <scope> 设 置 为 compi1e 估 会 把 AR 到 CLASSPATH 中 用 于 代码 的 编译 。 
将 <scope> 设 置 为 Lest 全 会 在 Maven 编 译 和 运行 测 起 时 把 这 些 JAR 加 到 CLASSPATH 中 。 

但 你 怎么 知道 该 指定 什么 人 <roupId>. <artifactId> 和 <version>? 答案 是 搜索 Maven 
Central Repository ( http://search.maven.org/ )， 你 几乎 总 能 找到 答案 。 

如 果 找 不 到 合适 的 工件 ， 可 以 用 install:install-file 目 标 自己 手工 下 载 和 安装 插件 。 
这 里 有 个 安装 asm-4.0 RC1.jar 类 库 的 例子 。 

mvn insatall:install-file 

-Dtile=asm-4.0 RC1 .Jar 

-Dgroupld=0Org .OW2 .aBm 

-DartifactId=AaAsm 

-Dversion=4.0 RCl1 

=-Dpackaging=jar 


这 个 命令 运行 完 后 ,你 应 该 能 在 本 地 资源 库 的 SHOME/.m2/repository/org/ow2/asm/asm/ 
4.0_RC1/ 中 找到 安装 好 的 工件 ， 就 像 Maven 下 载 的 一 样 。 


F， Pp nF 1 . ; a a p= Wh Ee by f 了 由: 隔 村: 站 三， 本 加 上 
| ! t re 各 "i i [A F 

| 4 二 
||| . 
Me 


i . > 时 - 
了 fe a 1 ye i L 中 
mn " ET | 了 | Ff nn” op n ] 站 
A i | | YN | | 要 
nan 了 | 


Pear 你 只 是 为 自己 散 的 ， pp 在 体 做 委 梁 同 
事 共享 的 工件 时 也 面临 相同 的 问题 , 但 你 也 不 能 把 它 放 到 Maven Central 中 ( 因为 那 是 你 们 的 私 
有 代码 )。 
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用 二 进 制 工件 管理 器 可 以 解决 这 个 问题 ， 比 如 Nexus ( http://nexus.sonatype.org/ )。 工 件 管 
理 器 就 像 你 和 困 队 自 有 的 本 地 Maven Central， 人 外 界 无 法 访问 。 大 多 数 工件 管理 器 还 会 狂 存 
Maven Central 和 其 他 资源 库 ， 你 的 开发 团队 所 需 的 依 环 项 都 可 以 从 它 那 里 得 到 。 


环境 配置 是 要 搞 懂 的 最 后 一 部 分 POM 了 ， 它 可 以 有 效 处 

4. 环境 配置 

环境 配置 是 Maven 用 来 处 理 环境 化 ( 比如 UAT 有 女生 产 环境 之 间 的 构建 差异 ) 或 其 他 与 普通 构 
建 稍 有 不 同 的 构建 变 体 的 。java7developer 项 目 中 有 个 例子 , 其 中 一 个 环境 配置 会 关 团 编 详 兹 和 作 
废 和 警告 ， 如 代码 清单 12-4 所 示 。 


代码 清单 12-4 ”POM: 环境 配置 
<profiles> 
<profiles 
<id>ignore-compiler-warnings</id> 和 一 一 一 一 一 一 一 一 一 
“build> 才 该 环境 配置 的 rp 
<Dluglinss 
<Bluglin> 
<roupId>org apache ,maven.plugins</groupId> 
<artifactlidsmaven-compiler-plugin</artifactIid> 
VErSions2.3.2</VersBions 
<CONnf igqurations 
HOUrcesl .7T</SgOurces 
<target>1.7T</target> 
<BhowDeprecation>false</ShowDeprecations> 局 关闭 警告 
<ShowWarningssfalae</sahowWNarningss 
<fork>true</fork> 
<executable>s${(jdk.Javac.fullpath})</executables 
</configurations 
</plugin> 
/pluginas> 
</build> 
</profile> 
/profiles> 


在 执行 Maven 时 用 -PpP <ia> 可 以 指定 要 启用 的 环境 配置 ( 比如 mvn compile -P ignore- 
compile-warnings ) 需 , 在 这 个 环境 配置 被 激活 后 , 会 使 用 指定 的 编 诺 器 择 件 ， 作 上 废 警告 和 其 
他 编译 占 警 告 都 会 被 关闭 各. 

在 Introduction to Build Profiles ( 构建 环境 配置 介绍 ) 页 面 可 以 找到 更 多 关于 环境 配置 和 为 其 
他 环境 化 目的 如 何 使 用 它们 的 信息 (http'Wmaven.apache.org/guides/introductiomrintroduction-to- 
profiles.html2 ), 

终于 完成 了 java7developer 项 目的 pom.xml 文 件 之 旅 ， 你 是 不 是 已 经 迫不及待 地 想 构 建 
它 了 ? 


里 不 同 环境 下 的 构建 。 


OD 短 链接 : bttp://tcn/z17MyO1。 一 一 译 者 注 
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12.3.2 ”运行 示例 


希望 你 已 经 把 代码 下 载 下 来 了 。 你 会 在 其 中 看 到 一 些 pom.xml 文 件 ， 就 是 它们 控制 着 Maven 
构建 。 

你 会 在 这 一 节 中 经 历 最 常用 的 Maven 构 建 周期 目标 (clean、compile、test 和 install )。 
第 一 个 目标 就 是 清除 上 次 构建 遗留 的 所 有 工件 。 

1. 清除 

标 clean 会 删 掉 target 目 录 。 请 换 到 $BOOK _ CODE 目录 并 执行 clean 目 标 。 


ca $BOOK CODE 
mvn clean 


与 你 要 用 到 的 其 他 Maven 构 建 目标 不 同 , clean 不 会 自动 调用 。 如 果 想 清除 上 次 构建 的 工件 ， 
必需 手动 加 上 clean 目 标 。 

上 次 构建 遗留 的 残余 物 已 经 清除 了 ， 接 下 来 一 般 是 执行 编译 源码 的 构建 目标 。 

2. 编译 

目标 compile 用 pom.xml 文 件 中 的 编译 问 插 件 配 壬 编译 在 src/main/java 、src/main/scala 和 | 
src/main/groovy 下 的 源码 。 也 就 是 说 它 会 带 着 加 到 cLAassPATH 中 的 编译 作用 域 依 闽 项 执行 Java、 
scala 和 Groovy 编 译 器 ( javac、scalac 和 groovyc )。Maven 还 会 处 理 在 src/main/resources 上 日 录 
下 的 资源 文件 ， 确 保 它们 作为 编译 cCLASSPATH 的 一 部 分 。 

编译 后 的 类 最 终 会 出 现在 target/classes 目 录 下 。 请 执行 下 面 的 Maven 目 标 : 

myn compile 


compile 目 标的 执行 应 该 相当 快 ， 并 且 在 控制 器 中 应 该 有 类 似 下 面 这 种 输出 。 


[INFO] [properties:read-project-properties (execution: default}] 

[INFO] [groovy:generateStubs (execution: default |}] 

[INFO] Generated 22 Java Stubs 

[INFO] [resources:resources (execution: default-resourcesl|] 

[INEO] Using ‘UTF-8' encoding to copy filtered regsources. 

[INFO] Copying 2 resgources 

[INFO] [compiler:compile {execution: default-compile}] 

[INFO] Compiling 119 source files to 
C:\Projecta\workspace3.6\code\trunk\target \classes 

[INFO] [scala:compile {execution: default}] 

[INEFO] Checking for multiple versions of scala 

[INFO] includes = [**/* .scala,**/* .java,] 

[INFO] excludes = [【] 

[INFO] C:\Projecta\workspace3.6\code\trunk\sarc\main\java:;-1: info: compiling 

[INFO] C:\Projectea\workspace3.6\code\trunk\target\generated-sources\groovy- 
atuba\main:-1: info: compiling 

[INFO] CC:\Projecta\workaspace3.6\code\trunk\arc\main\groovy:-1: info: 
compiling 

[INFO] CC: Dt DF RN We info: compiling 

[INFO] Compiling 143 source files to 
Ci\Projects\workspace3.6\code\trunk\target\classes at 1312716331031 

[INFO] prepare-compile in 0 a 
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[INFO] compile in 12 8 

[INFO] [groovy:compile lexecution: default}] 

[INFO] Compiled 26 "Ry 全 下 号 身 多 各 

[INFO] -=--------------- ---------- ---------- ee “= 
[INFO] BUILD SUCCESSFUL 

[INFO] ====-============== 和 和 -ee 
[INFO] Total time: #43 Seconds 

[INFO] Finished at: Sun AUG 07 12:25:44 BST 2011 

[INFO] Final ey 33M/79M 

[INFO])] -=----- -= 


在 这 一 阶段 ，src/test/java、 wei 和 ae 录 下 的 测试 类 还 没 编译 。 尽管 专门 
有 一 个 test-compile 目 标 编译 这 些 类 ， 但 最 常见 的 方式 是 运行 test 目标 。 

3. 测试 

在 目标 test 中 能 见 到 Maven 构 建 周 期 的 实际 效果 。 在 你 要 求 Maven 运 行 测试 目标 时 ， 它 知道 
为 了 保证 test 目 标 成 功 运行 ， 需 要 把 前 面 的 所 有 构建 周期 目标 都 执行 一 下 (包括 coempile、 
test-compile 和 一 系列 其 他 目标 )。 

Maven 会 通过 神火 Surefire ) 插件 ， 使 用 pom.xml 文 件 中 的 测试 提供 者 (作为 测试 作用 域 的 
依赖 项 ， 此 例 中 为 JUnit ) 运行 测试 。Maven 不 仅 会 运行 测试 ， 还 会 产生 报告 文件 ， 测 试 完成 后 你 
可 以 分 析 这 些 报告 ， 以 便 对 失败 测试 展开 调研 ， 并 收集 测试 指标 。 

请 执行 下 面 的 Maven 命 令 : 

mvn clean test 


一 旦 完成 测试 类 的 编译 和 运行 ， 应 该 就 能 见 到 下 面 这 种 输出 。 


i com.java7developer .chapterll ,listing 11 3.TicketRevenueTest 

Tests run: 5, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 0 sec 
Running com.java7Tdeveloper.chapterll,listing 11 4.TicketRevenueTest 

Tests run: 5, Faillures: 0, Errors: 0, Skipped: 0, Time elapsed: 0 sec 
Running com.JavaTdeveloper .chapterll,listing 11 5.TicketTest 

Teasts run: 1, Failures: 0, Errors: 0, Skipped: 0, Time elapaed: 0.015 Bec 


Resultes : 

Teaste Fun 20, Failures: 0, Errors: 0, Skipped: 0 
上 TE 
[INFO] BUILD SUCCESSFUL 


[INFO] Total time: 16 seconds 

[INFO] Finished at: Wed Jul 06 13:50:07 BST 2011 

[INFO] Final Memory: 24M/SB8M 

[INFO] =-------------- 


测试 结果 存在 target/surefire-reports 目 录 下 。 现 在 你 可 以 去 看 看 那里 的 文本 文件 。 稍 后 你 能 在 
一 个 更 棒 的 Web 页 面 上 看 到 这 些 结果 。 


提示 你 应 该 注意 到 了 ， 命令 中 还 有 clean 目 标 。 我 们 这 么 做 是 出 于 习惯 ， 以 防 有 扯 留 的 残余 
物 欺 骗 我 们 。 
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现在 你 的 代码 编译 过 ,也 测试 过 了 .并 且 已 经 准备 好 打包 了 。 尽 管 你 可 以 直接 用 package 
目标 ， 但 我 们 会 用 instal1 目 标 。 想 知道 为 什么 ， 且 看 下 文 分 解 ! 

4. 安 沪 

目标 install 的 任务 主要 有 两 个 , 按 pom.xml 文 件 中 <packaging> 指 定 的 方式 ( 此 例 中 为 JAR 
文件 ) 对 编译 结果 打包 。 然 后 把 打包 好 的 工件 安装 到 本 地 Maven 资 源 库 中 (在 SHOME/ 
.m2/repository 下 )， 以 便 其 他 项 目 可 以 把 它 当 做 依赖 项 用 。 跟 其 他 目标 一 样 ， 如 果 它 发 现 之 前 的 
构建 步骤 还 设 执 行 ， 它 也 会 先 执 行 这 些 相关 目标 。 请 执行 下 面 的 Maven 命 令 : 


mvn install 
一 旦 完成 install 目 标 ， 你 应 该 见 到 下 面 这 种 输出 报告 。 


[INFO] (jar:jar (execution: default-jar}) 

[INFO] Building jar: C:\Projecta\workapace3.6\code\trunk\target\ 

java7Tdeveloper-1.0.0.jar 

[INFO] [install:install {execution: default-install}] 

[INFO] Installing C:\Projectae\workspace3.6\code\trunk\target\java7Tdeveloper- 
1.0.0.ar 

to C:\Documents and Settings\Admin\ .m2\repository\com\jJava7Tdeveloper\ 

javaTdevelope 

r\l1.0.0\java7Tdeveloper-1.0.0.,jar 

[INFO] - =-------—=------=-----=----- -2 

[INFO] BUILD SUCCESSFUL 


[INFO] Total time: 17 seconds 

[INFO] Finished at: Wed Jul 06 13;:53;04 BST 2011 

[INFO] Final Memory: 28M/é6éM 

[INFO] = = =- 一 == 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 天 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 = 


在 target 目 录 ( package 目 标的 结果 ) 和 本 地 地 Maven 次 源 库 | 的 SHOME/.m2/repository/com. 
java7developer/1.0.0 目 录 下 应 该 能 找到 java7developer-1.0.0.jar 工 件 。 


提示 。 你 可 能 希望 把 Scala 和 Groovy 代 码 分 别 放 到 它们 自己 的 JAR 文 件 中 。Maven 支 持 这 种 操作 ， 
但 你 必须 记 住 对 于 Maven 来 说 ,每 个 独立 的 JAR 工 件 都 应 该 对 应 一 个 独立 的 项 目 。 也 就 
是 说 你 必须 用 到 Maven 中 多 模块 项 目的 概念 。 请 和 参见 Maven 的 Guide to Working with 
Multiple Modules ( 多 模块 处 理 指 南 ) 页 面 了 解 细 节 ( http://maven.apache.org/guides/mini/ 
guide-multiple-modules.html” ), 


我 们 大 多 数 人 都 是 在 团队 中 工作 , 并 且 经 常 要 共享 代码 库 , 那 我 们 怎么 才能 保证 快速 、 可 车 
地 构建 大 家 共享 的 代码 呢 ? 这 要 靠 CI 服 务 器 来 保证 ， 并 且 到 目前 为 止 ， 对 于 Java 开 发 人 员 来 说 现 
在 最 流行 的 CI 服务 器 是 Jenkins。 


中 短 链 接 http:/tcnjzjIIX70。 一 一 译 者 广 
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12.4 Jenkins: 满足 CI 需求 
CI 的 成 功 要 靠 管理 ( 开发 纪律 ) 和 工具 相 结合 。 为 了 让 CI 过 程 达 到 优秀 的 标准 ，Jenkins 提 供 
了 很 多 必需 的 支持 ， 如 表 12-2 所 示 。 


表 12-2 衡量 CI 构建 是 香 1 秀 的 标准 及 Jenkins 如 何 达成 这 些 标 准 


标 准 Jenkins 如 何 达成 
自动 构建 Jenkins 会 在 你 需要 的 任何 时 间 运 行 构建 。 它 可 以 通过 构建 般 发 器 实现 自动 构建 
一 直 测 斌 jenkins 能 运行 任何 你 想 要 的 目标 , 包括 Maven 的 test。 它 有 强大 的 测试 失败 趋势 报告 ， 只 要 
有 一 个 测试 没 通过 ， 它 就 会 报告 构建 失败 
定期 提交 这 是 开发 人 员 的 事 
每 次 提交 都 构建 。 ”每 次 检测 到 版 本 控制 库 的 新 提交 时 Jenkins 都 可 以 执行 构建 


快速 构建 这 对 基于 单元 测试 的 构建 更 加 重要 ,因为 你 想 要 它们 有 更 快 的 往返 时 间 。Jenkins 可 以 把 工作 
发 送 给 从 属 节点 从 而 提高 速度 ,但 更 主要 的 是 开发 人 员 做 出 精益 的 、 有 意义 的 构建 脚本 ， 并 
配置 Jenkins 在 执行 构建 时 调用 恰当 的 构建 周期 日 标 

结果 可 袖 化 Jenkins 有 基于 Web 的 仪表 板 ， 还 有 一 套 发 送 通 知 的 办 法 


所 有 CI 服务 器 都 能 轮 询 版 本 控制 资源 库 ， 并 执行 构建 周期 目标 compiLle 和 test。 让 Jenkins 
脱颖而出 的 是 它 易于 使 用 的 UI 和 可 扩展 的 插件 生态 系统 。 

在 配置 Jenkins 和 它 的 插件 时 ，UI 的 帮助 非常 大 , 它 经 常会 在 你 输入 完成 后 用 Ajax 检查 输 人 的 
有 效 性 。 它 还 提供 了 大 最 的 情景 式 帮 助 信 息 ， 运 行 Jenkins 根 本 就 不 需要 专业 技能 。 

Jenkins 的 插件 包罗 万 象 , 几乎 可 以 轮 询 任何 版 本 控制 资源 库 , 并 且 可 以 生成 一 系列 非常 有 价 
值 的 代码 报告 。 
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\ a ep Jenkine: 际 
现 的 一 个 副本 ， 主流 的 开发 人 员 和 活路 的 社区 现在 都 集中 在 Jenkins | 
个 优秀 的 CI 服务 器 ， 但 相 较 而 言 Jenkins 项 目 更 活跃 。 


Jenkins 是 自由 的 开源 软件 ， 其 社区 充满 活力 ， 对 新 手 帮 助 很 大 。 
关于 如 何 下 载 和 安 妆 Jenkins， 请 参阅 附录 D。 完 成 下 载 和 安装 后 ， 马 上 回来 继续 ! 


警告 假定 你 会 把 Jenkins 的 WAR 文 件 装 到 Web 服 务 器 上 ， 那么 Jenkins 安 装 的 根 URL 是 http 1 
alhost:8080/jenkins/, 如 条 是 直 搂 运行 WAR 文 件 ， 和 py /llocalhost:8080/, 


本 入 会 讨论 Jenkins 安 装 的 基础 配置 ,然后 是 如 何 设 置 .执行 构建 任务 。 pe 


中 指 运行 java -jar Jenkins.war。 一 一 幸 者 注 
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项 目 为 例 ， 但 你 可 以 随意 选用 自己 喜欢 的 项 目 。 
为 了 让 Jenkins 监 测 源码 资源 库 并 执行 构建 ， 需 要 先 配置 基础 设置 。 


12.4.1 基础 配置 


我 们 会 从 Jenkins 的 主页 http://localhost:8080/jenkins/ 开 始 。 要 配置 Jenkins， 请 点 击 左 边 菜单 中 
的 Manage Jenkins ( 管理 Jenkins ) 链接 ( http://localhost:8080/jenkins/manage )。 管 理 页 中 列 出 了 很 
多 设置 选项 。 

现在 ， 选 择 Configure System ( 系统 配置 ) 链接 ( http://localhost:8080/jenkins/configure )。 你 
应 该 能 进入 类 似 于 图 12-2 的 界面 中 。 


Jenkins 
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图 12-2 Jenkins 配 置 页 


从 界面 的 顶部 可 以 看 到 Jenkins 的 home 目 录 的 位 置 。 如 果 需 要 在 UI 之 外 进行 配置 ， 可 以 到 这 
个 目录 中 去 。 


提示 如果 你 是 为 团队 安装 Jenkins， 并 且 需 要 考虑 安全 性 ， 应 该 选中 Enable Security ( 安全 保护 ) 
和 Prevent Cross Site Request Forgery Exploits ( 阻止 跨 域 攻击 请 求 ) 多 选 框 ， 并 进行 相应 
的 配置 .对 初学 者 来 说 ,用 Jenkins 自 身 的 数据 库 最 容易 .以 后 你 可 以 随时 切换 到 企业 LDAP 
上 ， 或 基于 Active Directory ( 活动 目录 ) 进行 认证 和 授权 。 


为 了 执行 构建 ，Jenkins 震 要 知道 构建 工具 放 在 哪里 。 这 还 要 在 配置 页 中 设置 ， 找 到 单词 
“Maven 。 

1. 构建 工具 配置 

Jenkins 内 置 了 对 Ant 和 Maven ( 可 以 用 插件 支持 其 他 构建 工具 ) 的 支持 。 在 java7developer 项 
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目 中 ， 我 们 用 的 是 Maven ( 在 Windows 上 )， 所 以 对 Jenkins 的 配置 如 图 12-3 所 示 。 


Mven 
Maven 
Narne 
MAVEN_HOME |e 
Maven Installoetions 
TInstall Sutormatically [2 
bal Bl rn | 


总 二 本 人 晤 二 而 | 


LinE of Miwon install isns di hi 
图 12-3 设置 构建 工具 Maven 


注意 ，Jenkins 有 一 个 自动 安 妆 Maven 的 选项 ,在 没 村 Maven 的 机 化 上 ， 这 个 选项 还 是 挺 方便 的 。 

现在 Maven 配 置 好 了 ， 和 需要 告诉 Jenkins 你 用 什么 版 本 控制 资源 库 。 这 在 配置 页 的 下 面 。 找 到 
单词 SVN。 

2. 版 本 控制 配置 

Jenkins 内 置 了 对 CVS 和 Subversion ( SVN ) 的 支持 。 像 Git 和 Mercurial 这 样 的 版 本 控制 系统 也 
有 插件 。java7developer 项 目 用 SVN 1.6， 配 置 如 图 12-4 所 示 。 
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图 12-4 ”SYN 版 本 控制 配置 


在 设置 好 这 些 配置 后 ， 点 击 屏幕 底部 的 Save 按 钮 ， 以 确保 这 些 配 置 会 被 保存 下 来 。 
现在 Jenkins 的 基础 设置 已 经 做 好 了 ， 可 以 创建 你 的 第 一 个 任务 了 。 


12.4.2 ”设置 任务 

要 设置 新 任务 ， 请 回 到 仪表 板 中 并 点 击 左手 菜单 的 New Job ( 新 任务 ) 链接 ， 进 入 任务 设置 
页 面 ( http://localhost:8080/jenkins/view/All/newJob )。 这 里 有 很 多 选项 可 供 选 择 。 

要 设置 一 个 任务 来 构建 java7developer 项 目 ， 先 要 给 任务 确定 一 个 标题 ( java7developer )， 选 
择 Build a Maven 2/3 Project ( 构建 一 个 Maven 2/3 项 目 ) 选项 并 点 击 OK 按钮 继续 。 你 应 该 会 进入 
一 个 类 似 图 12-$ 的 配置 界面 。 这 里 有 一 些 输入 项 要 填 ， 但 下 面 这 些 应 该 是 你 先 填 好 的 内 容 : 
口 源码 管理 ; 
口 构建 触发 器 ; 
口 构建 。 
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图 12-5 Maven 2/3 任 务 配置 页 面 
我 们 从 源码 管理 的 配置 开始 。 
1. 源码 管理 
源码 管理 主要 设置 要 构建 的 源码 来 自 版 本 控制 的 哪个 分 支 、 标 记 或 标签 。 随 着 你 的 团队 向 版 
本 控制 系统 中 稳步 添加 源码 , 它 就 是 持续 集成 中 的 “集成 "。 对 于 java7developer 项 目 , 我 们 用 SVN 
构建 主干 中 的 源码 。 设 置 如 图 12-6 所 示 。 
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图 12-6 java7developer 源 码 管理 配置 


一 旦 告诉 Jenkins 从 哪里 获取 源码 , 接 下 来 要 配置 的 就 是 Jenkins 应 该 隔 多 长 时 间 构 建 一 次 ,这 
是 通过 构建 触发 器 完成 的 。 

2. 构建 触发 器 

构建 触发 硕 把 “持续 ”引信 了 持续 集成 。 你 可 以 要 求 Jenkins 在 源码 控制 库 每 次 有 新 提交 时 就 
进行 构建 ， 或 者 采用 更 悠 困 的 方式 ， 设 为 每 日 构建 一 次 。 

我 们 对 java7developer 项 目的 设置 , 只 是 要 求 Jenkins 每 隔 1$ 分 钟 轮 询 SVN 一 次 , 如 图 12-7 所 示 。 
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图 12-7 ”java7developer 构 建 触 发 器 配置 


你 可 以 点 击 输入 项 旁边 的 帮助 图 标 ( 表示 为 ? ) 查 看 帮助 信息 。 在 这 个 例子 中 , 编写 类 似 cron 
的 表达 式 来 指定 轮 鹿 周期 时 你 可 能 需要 帮助 。 

到 这 个 阶段 ，Jenkins 已 经 知道 了 到 哪里 去 找 源码 ， 隔 多 长 时 间 构 建 一 次 。 接 下 来 就 该 告诉 
Jenkins 详 该 执行 哪个 构建 阶段 ( 构建 脚本 中 的 目标 或 目的 )。 

3. 构建 

用 Jenkins 可 以 设置 很 多 任务 来 执行 构建 周期 的 不 同 阶段 ,你 可 能 想 要 一 个 每 晚 执行 一 次 完整 
的 系统 集成 测试 的 任务 。 但 更 多 情况 下 , 你 可 能 想 要 执行 频率 更 高 的 任务 , 在 每 次 有 新 的 源码 提 
交 到 版 本 控制 系统 时 编译 源码 并 运行 单元 测试 。 

对 于 java7developer 项 目 ， 我 们 要 求 Jenkins 执 行 Maven 的 clean 和 install 目 标 ， 如 图 12-8 
所 示 。 
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图 12-8 Java7developer 任 务 中 要 执行 的 Maven 构 建 目 标 (clean、instal1l ) 


Jenkins 现 在 有 java7developer 项 目的 所 有 信息 了 , 它 可 以 每 隔 15 分 钟 轮 询 一 次 SVN 代 码 库 的 主 
十 ， 执 行 Maven 的 clean 和 install 目 标 。 不 要 万 了 点 击 Save 按 钮 把 任务 存 下 来 | 

现在 可 以 回 到 仪表 板 ， 在 那里 看 看 你 的 任务 ， 它 应 该 非常 像 图 12-9。 

在 Last Success (S) 一 栏 ( 最 近 一 次 成 功 ), 圆 形 图 标 表示 任务 最 后 一 次 构建 的 状态 。 在 Weather 
(W) (天 气 ) 一 柱 ， 天气 图 标 表示 项 目的 总 体 健康 状况 , 它 是 由 构建 失败 的 频率 、 测 试 是 否 通过 ， 
还 有 一 系列 其 他 可 能 情况 ( 取决 于 你 配置 的 插件 ) 决定 的 。 要 进一步 了 解 这 些 图 标的 含义 ,请 点 
击 仪 表 板 中 的 Legend ( 图 例 ) 链接 ( http://localhost:8080/jenkins/legend )。 

现在 任务 已 经 准备 好 了 , 你 可 能 想 看 看 它 运 行 起 来 是 什么 样 ! 你 可 以 等 15 分 钟 后 的 第 一 次 轮 
测 ， 也 可 以 强制 执行 一 次 构建 。 
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图 12-9 有 java7developer 任 务 的 仪表 板 


12.4.3 执行 任务 


要 检查 新 配置 ,强迫 任务 执行 是 个 好 办 法 。 对 于 java7developer 任 务 ， 只 要 到 仪表 板 中 ， 点击 
Schedule a Build ( 调度 构建 ) 按钮 ( 在 Last Duration 旁 边 带 绿色 箭头 的 钟表 图 标 )。 然 后 你 就 能 刷 
新 页 面 去 看 看 构建 的 执行 结果 了 。 


提示 点击 仪表 板 右 上 角 的 Enable Auto Refresh ( 允许 自动 刷新 )， 可 以 让 仪表 板 自 动 剧 新 。 这 
样 你 就 可 以 及 时 看 到 当前 构建 的 状态 了 。 

在 构建 执行 时 ，java7developer 任 务 的 第 一 个 图 标 会 闪 , 表明 构建 正在 处 理 中 。 在 页 面 左 侧 还 
能 看 到 Build Executor Status ( 构建 执行 着 状态 )。 构建 一 完成 ，Last Success (S) (最近 一 次 成 功 ) 
一 栏 那个 圆 形 图 标 就 变 成 了 红色 ， 表 明 构 建 失败 了 。 

这 个 失败 是 因为 漏 掉 了 build.properties 文 件 。 如 果 你 阅读 12.2 节 时 没 按照 书 中 的 指示 做 , 现在 
也 可 以 很 快 解 决 掉 这 个 问题 ， 找 到 build.properties 文 件 样本 ， 编 辑 它 ， 以 便 能 找到 要 用 到 的 Java7 
JDK。 下 面 是 在 Unix 上 执行 这 些 操作 的 例子 : 


cd $USER/ .jenkins/jobs/ijavaTrdeveloper/workspace/iava7Tdeveloper 
cp sample build unix.properties build.properties 


现在 你 可 以 回 到 仪表 板 中 ， 再 次 手工 运行 构建 。 这 次 构建 应 该 能 成 功 ， 并 且 仪 表 板 中 
java7developer 任 务 的 Last Success ( 最 近 一 次 成 功 ) 一 栏 应 该 是 个 蓝 色 图 标 ， 表 示 构 建成 功 了 。 

你 还 可 以 马上 去 看 一 下 构建 的 测试 报告 ， 因 为 Jenkins 知 道 如 何 读 取 Maven 产 生 的 输出 。 要 看 
测试 结果 ， 可 以 点 击 java7developer 任 务 Last Success 一 栏 的 链接 ( http://localhost:8080/jenkins/job/ 
java7developer/lastSuccessfulBuild/ )。 该 链接 会 将 你 带 人 Latest Test Result ( 最 新 测试 结果 ) 页 面 ， 
其 界面 如 图 12-10 所 示 。 
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图 12-10 java7developer 成 功 构建 的 测试 结果 


测试 全 部 通过 ， 非 常 棒 ! 如 果 有 任何 一 个 失败 ， 你 可 以 深入 了 解 每 个 测试 的 细节 。 

我 们 对 运行 失败 和 成 功 构建 的 基础 都 进行 了 总 结 。 对 于 java7developer 项 目 来 说 , Jenkins 会 继 
续 轮 询 SVN 并 在 发 现 新 提交 时 执行 新 构建 。 

你 已 经 见 过 Jenkins 如 何 运 行 构建 , 构建 因 不 同 原因 失败 时 如 何在 界面 上 发 出 警告 如 何 检 查 
测试 成 功 或 失败 。 但 Jenkins 可 以 做 得 不 止 这 些 , 它 还 能 给 出 一 系列 实用 的 代码 指标 ， 让 你 对 代码 
的 质量 有 更 深入 的 认识 。 


12.5 ” Maven 和 Jenkins 代码 指标 


Java 和 JVM 已 经 存在 很 长 时 间 了 ， 并 且 这 么 些 年 积累 下 来 ， 已 经 开发 出 了 一 些 强大 的 工具 和 
类 库 来 指导 开发 人 员 编 写 优质 的 代码 。 我 们 宽泛 地 把 这 个 领域 定义 为 代码 指标 或 静态 代码 分 析 。 
Maven 和 Jenkins 都 支持 目前 最 流行 的 工具 。 尽 管 这 些 流行 的 工具 (或 新 的 专门 化 工具 ) 对 其 他 语 
言 的 支持 越 来 越 多， 但 它们 仍然 主要 是 针对 Java 语 言 自身 。 


提示 “现代 IDE ( 比如 Eclipse、IntelliJ 和 NetBeans ) 也 支持 几 种 静态 语言 分 析 工 具 和 类 库 ， 值 得 
花 些 时 间 研 究 一 下 。 


代码 指标 工具 主要 是 为 了 消除 所 有 开发 人 员 都 会 犯 的 小 错误 。 它 能 帮 你 树 起 最 低 的 代码 质量 
标杆 ， 告 诉 你 下 面 这 些 情况 :; 

口 测试 对 代码 的 覆盖 率 ”; 

口 代码 的 格式 是 否 清晰 ( 有 助 于 差异 比较 和 可 读 性 ); 

口 是 否 很 可 能 会 出 现 NPE; 

口 是 否 态 记 了 域 对 象 中 的 equals () 和 hashcode() 方 法 。 


册 本 书 不 讨论 普通 的 代码 覆盖 工具 ， 因 为 它们 和 Java 7 还 不 兼容 。 
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各 种 工具 提供 的 检查 列表 都 很 长 ， 但 开发 团队 要 自己 决定 对 项 目 做 哪些 检查 。 


a 一 些 团 从 因为 解决 了 代码 指标 工具 区 各 过 约 问题 就 认为 他 们 的 代 三 库 妹 于 完善 了 ， 这 是 不 
对 的 。 代 码 指 标 警 告 能 帮 你 去 掉 很 多 低级 bug， 避 免 糟糕 的 编码 实践 。 但 它们 不 能 保证 代码 ， 
量 , 或 者 判断 业务 逻辑 实现 是 否 正 确 。 

另外 一 个 问题 是 管理 层 可 能 想 把 这 些 指标 放 到 报告 里 。 为 了 管理 层 和 你 们 自己 考虑 ， 请 把 
代码 指标 留 在 开发 人 员 这 一 层面 。 它 们 不 是 项 目 管 理 指标 。 


Maven 和 Jenkins 结 合 得 很 好 ， 既 可 以 提供 代码 指标 的 概览 ， 也 能 告诉 你 其 中 的 细节 。 本 节 的 
两 个 主要 内 容 如 下 ， 

口 如 何 安装 并 配置 Jenkins 插 件 ; 

口 如 何 配置 代码 一 致 性 ( Checkstyle ) 和 bug 查 找 ( FindBuss ) 插件 。 

我 们 仍 以 java7developer 为 例 。 先 来 看 看 如 何 安 装 Jenkins 捕 件 , 这 是 得 到 代码 指标 报告 功能 的 
前 提 条 件 。 
12.5.1 安装 Jenkins 插件 

Jenkins 基 于 UI 的 插件 管理 器 很 不 错 , 它 可 以 帮 你 下 载 和 安装 捅 件 , 所 以 安装 Jenkins 插 件 并 不 
复杂 。 安 装 Jenkins 捕 件 时 需要 重启 Jenkins， 所 以 应 该 先进 入 Jenkins 管 理 页 ( http://localhost: 
8080/jenkins/manage )， 并 点 击 Prepare to Shutdown ( 准备 关闭 ) 链接 。 这 会 终止 所 有 即将 执行 的 
任务 ， 以 便 你 可 以 安全 地 安装 插件 ， 重 启 Jenkins。 

Jenkins 准 备 好 头 财 之 后 ， 就 可 以 访问 捅 件 管理 带 了 。 在 管理 页 中 点 击 Manage Plugins ( 管理 
插件 ) 链接 ( http://localhost: OR )。 应 该 能 见 到 如 图 12-11 所 示 的 界面 。 


Je TT 


图 12-11 Jenkins 择 件 管理 器 
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第 一 个 是 Updates ( 更 新 ) 标签 。 切 换 到 Available ( 可 用 ) 标 等， 能 见 到 一 长 串 可 用 插件 的 列 
表 。 为 学 习 本 章 内 容 ， 需 要 选中 下 面 这 些 插件 : 

口 Checkstyle; 

口 FindBugs。 

然后 点 击 界面 底部 的 Install 按 钮 开始 安装 。 安 装 完 成 后 ， 可 用 通过 链接 http://localhost:8080/ 
jenkins/restart 生 局 Jenkins。 

重启 Jenkins 之 后 插件 就 安装 好 了 。 现 在 该 去 配置 这 些 插 件 了 ， 先 从 Checkstyle 插 件 开始 。 


12.5.2 ”用 Checkstyle 保持 代码 一 致 性 


Checkstyle 是 静态 代码 分 析 工 具 ， 主 要 关注 代码 布局 、Javadoc 层 次 是 否 恰当 ， 以 及 其 他 语法 
糖 的 检查 。 它 还 会 检查 常见 的 代码 错误 ， 但 FindBugs 检 查 得 更 加 全 面 。 

Checkstyle 的 重要 性 体现 在 两 个 方面 。 首 先 ， 它 有 助 于 强化 小 组 的 编码 风格 规范 ， 以 便 团 队 
成 员 可 以 很 容易 地 读 懂 彼此 的 代码 ( 易 读 性 是 Java 得 以 流行 的 一 个 主要 原因 )。 其 次 ， 如 果 代 码 
元 素 的 位 置 和 空格 保持 一 致 ，diffs 和 patches 用 起 来 就 更 容易 了 。 

我 们 在 Maven 的 pom.xml 文 件 中 已 经 配置 过 Checkstyle 捅 件 了 , 所 以 你 只 需要 在 java7developer 
任务 中 加 上 checkstyle:checkstyle 日 标 。 要 配置 该 任 务 点 击 在 仪 表 板 中 列 出 的 
java7developer 链 接 ， 然 后 在 新 界面 上 点 击 左 侧 某 单 中 的 Configure (配置 ) 链接 。 

接着 配置 报告 ， 并 确定 违规 的 情况 出 现 次 数 太 多 时 是 否 应 该 放弃 构建 。 图 12-12 中 是 我 们 对 
java7developer 项 目 中 Maven 构 建 及 报告 的 配置 。 


图 12-12 Checkstyle 配 置 


别 忘 了 点 击 Save 把 这 个 配置 存 下 来 ! Checkstyle 的 默认 规则 是 Sun 公 司 最 初 提 出 来 的 Java 编 码 
规范 。Checkstyle 能 微调 到 第 n 级 ， 所 以 应 该 能 准确 表示 团队 的 编码 规范 。 


警告 ”了 荫 新 版 的 Checkstyle 还 没 全 面 支持 Java 7 语法 ， 所 以 你 见 到 对 try-with-resources、 和 钻石 语法 
和 其 他 Coin 项 目 语法 的 肯定 可 能 是 假 的 。 
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让 我 们 来 看 看 默认 规则 集 如 何 把 Java7developer 项 目 组 织 起 来 。 跟 往常 一 样 ， 你 可 以 回 到 
Jenkins 的 仪表 板 中 ， 手 工 执行 构建 。 构 建 完成 后 ， 回 到 最 近 构 建成 功 页 面 ( 记 住 ， 可 以 通过 Last 
Success 一 栏 的 链接 访问 该 页 面 )， 点 击 左 侧 菜单 上 的 Checkstyle Warnings ( Checkstyle 普 告 ) 链接 
进入 Checkstyle 报 告 页 。java7developer 项 目的 Checkstyle 报 告 页 看 起 来 应 该 如 图 12-13 所 示 。 
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图 12-13 ”Checkstyle 报 告 


如 你 所 见 ， 在 Java7developer 的 代码 上 有 些 和 警告 信息 。 看 起 来 我 们 还 有 些 工 作 要 做 ! 你 可 以 
深入 到 每 个 警告 中 ， 看 看 为 什么 会 发 生 违规 ， 并 在 下 一 次 构建 之 前 改正 它 。 

Checkstyle 肯 定 在 这 方面 有 所 帮助 ， 但 它 的 重点 不 是 潜在 的 代码 错误 。 这 种 重要 的 代码 错误 
检查 最 好 用 FindBugs 捅 件 。 


12.5.3 用 FindBugs 设 定 质量 标杆 


FindBugs ( Bill Pugh 出 品 ) 是 为 了 找 出 代码 中 潜在 的 bug 而 做 的 字 节 码 分 析 工 具 。 由 于 它 分 
析 的 是 字 节 码 , 所 以 也 能 用 在 Scala 和 Groovy 上 。 但 因为 它 所 设置 的 规则 是 为 了 捕获 Java 语 言 中 的 
bug， 所 以 如 果 你 用 它 来 分 析 Groovy 和 Scala 代 码 ， 要 对 其 持 审 慎 的 态度 ， 因 为 它 可 能 发 现 不 了 其 
中 的 bug。 

FindBugs 背 后 有 大 量 人 研究 成 果 的 支持 ， 都 是 由 特别 熟悉 Java 语 言 的 开发 人 员 做 的 。 它 能 检测 
出 下 面 这 些 状况 : 

口 会 导致 NPE 的 代码 ; 

口 赋值 给 一 个 从 来 设 用 到 的 变量 ; 

口 用 == 而 不 是 用 eauals 方 法 比较 宁 竺 串 对 象 ; 

口 在 人情 环 中 用 基本 的 + 操作 符 而 不 是 用 stringBuffer 合 并 字符 审 

你 应 该 先 试 试 FindBugs 的 默认 设置 ， 然 后 再 根据 要 检测 的 规则 进行 微调 。 


了 
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警告 ”即便 是 Java 语 言 ，FindBugs 也 会 误 报 。 你 应 该 认真 研究 它 发 出 的 警告 ， 如 果 确 信 可 以 息 略 
它们 ， 可 以 显 式 排除 这 些 情况 。 

FindBugs 的 重要 性 体现 在 两 个 方面 ,首先 , 它 以 结对 程序 员 的 角色 教 开 发 人 员 养 成 好 习惯 ( 尽 
可 能 帮助 检测 出 潜在 bug )。 其 次 , 项 目的 总 体质 量变 好 了 ,并 且 问 题 跟踪 单 里 不 会 再 充满 烦人 的 
小 bug， 让 团队 可 以 解决 实际 问题 ， 比 如 业务 逻辑 的 变化 。 

跟 Checkstyle 插 件 一 样 ， 你 可 以 点 击 仪 表盘 任务 列表 中 的 java7developer 和 链接 ， 人 然后 在 新 界面 
中 点 击 左 侧 菜单 上 的 Configure 链 接 。 

为 了 执行 FindBugs 捅 件 ， 还 要 在 Jenkins 中 添加 Maven 构 建 目标 compile findbugs:findbugs 
(需要 compile 以 使 FindBugs 能 处 理 字 市 码 )。 

除了 定义 违例 过 多 构建 是 否 失败 ， 还 可 以 配置 报告 。 如 图 12-14 所 示 。 
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图 12-14 FindBugs 配 置 


别 筷 了 点 击 Save 把 这 个 配置 存 下 来 ! FindBugs 预 定义 的 规则 集 可 以 大 范围 调整 ， 以 准确 表示 
团队 的 编码 规范 。 让 我 们 来 看 看 黑 认 规则 集 如 何 应 用 到 Java7developer 项 目 上 。 

跟 往 常 一 样 ， 你 可 以 回 到 Jenkins 的 仪表 盘 中 , 手工 执行 构建 。 构 建 完 成 后 ， 回 到 最 近 构 建成 
功 页 面 ( 记 住 , 可 以 通过 Last Success 栏 的 链接 访问 该 页 面 ), 点 击 左 侧 菜 单 上 的 FindBugs Wamings 
链接 进入 FindBugs 报 告 矶 。java7developer 项 目的 FindBugs 报 告 页 看 起 来 应 该 如 图 12-15 所 示 。 

如 你 所 见 ，Java7developer 的 代码 有 些 痪 告 信 息 。 写 书 的 作者 也 可 能 写 出 不 完美 的 代码 ! 你 可 
以 仔细 检查 每 个 警告 ， 看 看 为 什么 会 发 生 违 规 ， 并 且 如 果 你 愿意 ， 可 以 在 下 一 次 构建 之 前 改正 它 。 

FindBugs 会 把 大 部 分 亲 见 的 Java 陷 阱 和 编码 错误 都 找 出 来 。 随 着 开发 团队 对 这 些 错误 的 了 解 
程度 不 断 加 这 ,报告 中 的 警告 数量 会 逐步 减少 。 你 们 不 仅 提升 了 代码 的 品质 ,还 完善 了 自身 的 编 
码 能 力 ! 
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图 12-15 ”FindBugs 报 告 页 
Jenkins 、Maven 和 代码 指标 这 一 节 到 此 就 完成 了 。 这 一 领域 的 工具 化 程度 相当 高 ( Scala 和 
Groovy 还 需要 更 多 支持 )， 启 动 和 运行 也 非常 容易 。 如 果 你 是 CI 迷 ， 想 探索 Jenkins 的 完整 能 力 ， 
我 们 强烈 推荐 John Ferguson Smart 不 断 更 新 的 Jenkins: The Definitive Guide( O'Reilly )。 你 可 能 已 
经 注意 到 了 ， 与 JYM 多 语言 编程 相关 的 构建 和 CI 的 拼图 还 缺 了 一 片 一 一 我 们 还 没 处 理 过 Clojure 
项 目 。 好 在 Clojure 社 区 已 经 专门 针对 纯粹 的 Clojure 项 目 制作 了 几 个 构建 工具 , 并 已 得 到 了 广泛 应 
用 。 其 中 最 流行 的 就 是 Leiningen， 一 个 专 为 Clojure 写 的 构建 工具 。 


12.6 Leiningen 


要 成 为 对 开发 人 员 有 用 的 构建 工具 ， 关 键 是 要 具备 下 面 这 几 种 能 力 : 

口 依赖 管理 ; 

口 编译 ; 

口 测试 自动 化 ; 

口 部 署 打 和 包 。 

Leiningen 对 此 采取 的 策略 是 分 而 治之 。 它 重用 已 有 的 Java 技 术 实现 了 每 种 功能 , 但 却 没有 把 
所 有 功能 都 放 在 一 个 包 里 。 

这 听 上 去 可 能 比较 复 末 ,还 有 点 仙 怖 ,但 开发 人 员 并 不 会 受到 这 种 复杂 性 的 影响 。 实 际 上 ， 
基 至 没 用 过 底层 Java 工 具 的 开发 人 员 也 能 使 用 Leiningen。 我 们 一 开始 先 通过 一 个 非常 简单 的 过 程 
来 安装 Leiningen。 然 后 讨论 Leiningen 的 组 件 和 整体 架构 ， 最 后 用 Hello World 项 目 来 小 试 牛刀 。 

你 将 看 到 如 何 开始 一 个 新 项 目 , 添加 依赖 项 , 使 用 Leiningen 提 供 的 Clojure REPL 内 部 依赖 项 。 
这 目 然 会 让 我 们 转 而 讨论 如 何 用 Leiningen 在 Clojure 内 做 TDD。 作为 本 章 的 收尾 , 我 们 会 看 一 下 如 
何 打包 代码 ， 产 生 一 个 应 用 程序 部 署 或 供 人 调用 的 类 库 。 

我 们 来 看 看 如 何 开始 使 用 Leiningen 吧 。 
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12.6.1 Leiningen 入 门 


Leiningen 非 常 容易 上 手 。 对 于 类 Unix 系 统 ( 包括 Linux 和 Mac OS XX), 开发 人 员 可 以 从 和 擎 握 lein 
脚本 开始 。 在 GitHub 上 可 以 找到 它 Leiningen (在 https://github.com/ 页 面 中 或 用 自己 喜欢 的 搜索 引 
擎 搜索 Leiningen )。 

把 lein 肢 本 放 到 PATH 中 并 设 为 可 执行 文件 后 ， 它 就 可 以 运行 了 。 在 第 一 次 运行 lein 时 ， 它 会 
检查 需要 安装 哪些 依赖 项 ( 还 有 哪些 已 经 装 上 了 )。 只 要 需要 ， 它 甚至 会 把 其 他 不 属于 Leiningen 


核心 部 分 的 组 件 也 给 装 上 。 因 为 要 安装 依赖 项 ， 首 次 运行 可 能 比 后 续 运 行 稍 慢 
在 下 一 节 ， 我 们 会 介绍 Leiningen 的 架构 ， 以 及 为 它 提供 核心 功能 的 Java 技 术 。 


rg ry 
生存 芍 < 标准 的 、 Mea 比如 说 ,基本 的 Windows 安 装 叶 TTTI 
- 具 ( Leiningen 需 要 用 它们 pr pe gen Wind' 
安装 - 带 有 cinbat 文 件 和 预 轩 的 wgstext 压 绽 件 ， 为 了 直 自 行 安装 的 lein 正 确 工 作 ， 需 把 
它们 放 到 Windows 的 PATH 中 的 目录 下 。 


12.6.2 Leiningen 的 架构 

我 们 说 过 , Leiningen 封 装 了 一 些 主流 的 Java 技 术 并 做 了 简化 。 它 封装 的 主要 组 件 是 Maven( 版 
本 2 )、Ant 和 javac。 

如 图 12-16 所 示 ，Maven 用 来 做 依赖 项 解析 和 管理 ，javac 和 Ant 用 来 构建 、 运 行 测试 和 完成 
构建 过 程 中 的 其 他 工作 。 


图 12-16 ” Leiningen 及 其 组 件 
高 级 用 户 可 以 穿 过 抽象 层 , 直接 使 用 Leiningen 的 底层 工具 。 但 Leiningen 的 基本 语法 和 应 用 非 


党 简单， 不 需要 使 用 者 具备 使 用 任何 底层 工具 的 经 验 。 
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我 们 来 看 一 个 简单 的 例子 , 看 看 project.clj 文 件 如 何 工 作 , 以 及 在 Leiningen 项 目 生命 周期 中 如 
何 使 用 那些 基本 的 命令 。 


12.6.3 Hello Lein 
把 lein 放 在 PATH 上 之 后 ， 我 们 可 以 用 它 的 new 命 令 开始 一 个 新 项 目 ， 


ariel:projects boxcat$ lein new hello-lein 

Created new project in: /Users/boxcat/projects/hello-lein 
ariel:projects boxcats$ cd hello-lein/ 

ariel:hello-lein boxcats ls 

README project .cl] src teast 


这 个 命令 创建 了 一 个 叫做 hello-lein 的 项 目 。 它 有 项 目 目录 , 里 面 有 个 简单 的 描述 文件 README、 
一 个 project.clj 文 件 ( 马上 就 会 详细 讨论 )， 还 有 并 列 的 srec 和 (test 目录 。 

如 果 你 把 Leiningen 刚 刚 创 建 的 项 目 导 人 Eclipse 中 ( 比如 用 CounterClockwise 捕 件 ), 项 目的 布 
局 应 该 如 图 12-17 所 示 。 


ve Helio Lein 
vsre 
™ EB hello_hein 
ON core .dl 
bp mh MRE System Library liava- 1.7.0-iniernal-mhvm-201 
>» Bh Referenced Libraries 
> Classes 
TT jt 
"Ey hello_lein 
TL test 
OD core .cl 
dojure-contrib-src.jar 
cdojure-contrib ,jar 
等 colure-srcjar 
二 ciolure jar 
WD project.dl 
3 README 


图 12-17 新 创建 的 Leiningen 项 目 


这 个 项 目 结构 是 直接 从 Java 项 目 上 照搬 过 来 的 ; 有 带 有 core.clj 文 件 的 并 列 test 和 src 结 构 (分 
别 用 于 测试 和 顶层 代码 )。 另 外 一 个 重要 的 文件 是 project.clj，Leiningen 用 它 来 控制 构建 、 保 存 元 
数据 。 

我 们 来 看 一 下 lein 的 new 命 令 生 成 的 骨架 文件 。 

(defproiject hello-lein "1.0.0-SNAPSHOT™" 


:description "FIXME: write description" 
:dependencies [[org.clojure/clojure "1.2.1"]]) 


这 个 Clojure 形 式 解 析 起 来 相当 直 日 : 有 一 个 (defproject) 的 宏 负 责 制 作 表 示 gen 机 
的 新 值 。 这 个 宏 需 要 知道 项 目 名 称 (在 这 里 是 hello-lein )， 还 需要 知道 项 目的 版 本 ( 黑 认 是 
1.0.0-SNAPSHOT，12.3.1 节 讨论 过 的 Maven 版 本 号 )， 然 后 是 描述 项 目的 元 数据 映射 。 


1 入 
9 


328 第 12 章 构建 和 持续 集成 


lein 自 带 了 两 个 元 数据 :一 个 描述 字符 串 和 一 个 依赖 项 向 量 ， 后 者 对 于 添加 新 的 依赖 项 很 方 
便 。 我 们 现在 就 来 加 一 个 clj-time 类 库 。 这 个 类 库 为 Clojure 提 供 Java 日 期 和 时 间 类 库 (Joda-Time， 
但 在 这 个 例子 中 你 没 必 要 知道 这 个 Java 类 库 ) 的 访问 接口 。 加 上 新 的 依赖 项 后 ，project.clj 看 起 来 
应 该 是 这 样 的 : 

(defproject hello-lein "1.0.0-SNAPSHOT" 

:description "FIXME: write description" 


:dependencies [[org.cloiure/cloijure "1.2.1"] 
[el1=time "0.3.0"™]])) 


向 量 的 第 二 个 元 素描 述 了 要 用 的 新 依赖 项 类 库 版 本 ,如 果 Leiningen 在 本 地 依赖 项 资源 库 中 找 
不 到 它 ， 会 按 这 个 版 本 从 外 部 资源 库 获 取 该 依赖 项 。 

Leiningen 默 认 从 位 于 http:Wcloijars.org/ 的 资源 库 获 取 缺 失 的 兴 库 。 因 为 Leiningen 撒 层 用 的 是 
Maven， 所 以 这 本 质 上 就 是 一 个 Maven 资 源 库 。Clojars 提 供 了 一 个 搜索 工具 ， 可 以 在 你 知道 所 需 
类 库 但 不 知道 具体 版 本 号 时 提供 帮助 。 

在 这 个 新 的 依赖 项 就 位 后 ， 你 需要 更 新 本 地 构建 环境 ， 可 以 执行 ]ein deps 命 令 。 


ariel:hello-lein boxcats lein deps 

Downloading: cli-time/clij-time/0.3.0/clj-time-0.3.0,pom from central 
Dovwnloading: clij-time/clij-time/0.3.0/clij-time-0.3.0.pom from cloijure 
Downloading: clj-time/clj-time/0.3.0/clij-time-0.3.0.pom from clojars 
Transferring 2K from cloijars 

Downloading: joda-time/joda-time/l.6/joda-time-1.6.pom from clojure 
Downloading: oda-time/joda-time/l.68/joda-time-1.6.pom from cloijars 
Transferring SK from cloars 

Downloading: clj-time/clij-time/0.3.0/clij-time-0.3.0.jar from central 
Downloading: clj-time/clj-time/0.3.0/cli-time-0.3.0.jar from cloijure 
Downloading: clij-time/clij-time/0.3.0/c1j-time-0.3.0.jar from clojars 
Tranaferring 7K from clojars 

Downloading: Joda-time/joda-time/l1.6/joda-time-1.6.Jar from clojure 
Downloading: joda-time/joda-time/l1.6/joda-time-1.6.jar from clojars 
Transferring S22K from clojars 

Copvying 4 files to /Users,/boxcat/projects/hello-lein,/lib 
ariel:hello-lein boxcats 


Leiningen 已 经 用 Maven 下 载 了 Clojure 的 接口 , 还 有 底层 的 Joda-Time JAR。 我 们 在 代码 中 用 一 
下 它 ， 展 示 在 依赖 项 存在 的 情况 下 如 何 用 Leiningen 作 为 REPL 进 行 开 发 。 
需要 把 主要 源 文 件 src/hello lein/core.clj 改 成 下 面 这 样 : 


Ins hello-lein .core) 


(use '[clj-time.core :only (ldate-time})]) 


Idefn isodate-to-millis-since-epoch [x] 
(.getMillis (lapply date-time 
= (map #(Integqer/parselInt $%) (.split x "-"))))) 


它 提 供 了 一 个 Clojure 函 数 , 将 ISO 标 准 日 期 (格式 为 YYYY-MM-DD ) 转 换 成 自 Unix 纪 元 ( 1970 
年 ) 以 来 的 毫秒 数 。 
我 们 用 Leiningen 的 REPL 风 格 测 试 一 下 。 先 在 project.clj 文 件 中 加 上 一 行 ， 改 成 下 面 这 样 : 
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(defproject hello-lein "1.0.0-SNAPSHOT" 
:description "FIXME: write description" 
dependencies [[org.clojure/clojure "1.2.1"] 

[elij-time "0,3.0"]] 
:repl=init hello-lein.core) 


加 上 这 一 行 后 ， 可 以 启动 一 个 能 访问 所 有 依赖 项 的 REPL， 并 且 它 已 经 把 命名 空间 hel1o- 
Lein.core 中 的 图 数 引 人 了 了 作用 域 ; 


ariel:hello-lein boxcats$ lein repl 
REPL gtarted; server listening on localhost:10886. 


hello-lein.core=s {isodate-to-millis-since-epoch "1970-01-02") 

86400000 

hello-lein.core=> 

这 是 以 天 为 单位 的 日 期 的 正确 毫秒 数 ， 并 且 它 阐明 了 在 真实 项 目 中 使 用 REPL 的 核心 原则 。 
我 们 在 这 上 面 稍微 展开 一 点 ， 青 看 一 个 使 用 Leiningen REPL 面 向 测试 的 工作 方式 。 


12.6.4 用 Leiningen 做 面向 REPL 的 TDD 


任何 优秀 的 TDD 方 法 ， 其 核心 都 应 该 是 一 个 用 来 开发 新 功能 的 简单 基本 的 循环 。 具 体 到 
Clojure 和 Leiningen， 其 基本 循环 应 该 如 下 所 示 : 

(1) 添加 任何 所 需 的 新 依赖 项 ( 并 重新 运行 ]ein deps ); 

(2) 局 动 REPL ( lein repl ); 

(3) 草拟 一 个 新 函数 ， 并 把 它 放 到 REPL 的 作用 域 中 来 ; 

(4) 在 REPL 内 测试 这 个 函数 ; 

(5) 重复 步骤 3 和 4， 和 直到 该 函数 表现 正确 ; 

(6) 把 该 函数 的 最 终 版 加 到 恰当 的 ,cj 文件 上 ; 

(7) 把 刚才 运行 的 测试 用 例 加 到 测试 集 .clj 文 件 中 ; 

(8) 重启 REPL， 再 次 从 第 三 步 开始 ( 或 者 第 一 步 ， 如 果 需 要 新 的 依赖 项 )。 

这 是 测试 驱动 开发 的 风格 , 但 却 避 开 了 先 写 测试 还 是 先 写 代码 的 问题 , 用 REPL 风 格 的 TDD， 
这 两 件 事 是 同时 进行 的 。 

之 所 以 要 在 添加 新 困 数 时 重启 REPL ( 第 八 步 ), 是 为 了 能 干净 地 编译 新 函数 , 创建 新 函数 时 ， 
有 时 为 了 支持 它 , 会 对 其 他 函数 或 环境 做 轻微 的 修改 。 而 这 些 修改 在 把 函数 加 入 最 终 的 源码 库 时 
很 容易 被 忘掉 。 重 启 REPL 能 帮 我 们 尽早 记 起 这 些 被 忘掉 的 修改 。 

这 个 过 程 清晰 而 简单 ,但 还 有 个 问题 我 们 没 提 到 ,无 论 是 在 这 里 还 是 第 11 章 讨论 TDD 时 , 我 
们 都 没 讨论 过 怎么 编写 Clojure 测 试 。 好 在 这 非常 简单 。 我 们 来 看 一 下 用 lein new 创 建新 项 目 时 
它 所 提供 的 模板 : 

(ns hello-lein.test .core 


(:use [hello-lein.core]) 
(suse [lclojure.test]))) 


[deftest replace-me ;; FIXME: write 
(is false "No tests have been written,")) 
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我 们 就 用 lein test 命 令 来 测试 这 个 自动 生成 的 用 例 ， 看 看 会 发 生 什么 ( 实际 上 你 应 该 能 猜 
出 来 ) 

ariel :hello-lein boxcats lein test 

Testing hello-lein,.test .core 


FAIL in (replace-me) (Core .cl]:6) 
NO tests have been vritten. 
axpectad: false 

actual: false 
Ran 1 tests containing 1 assertions. 
1 tailures, 0 errors. 


如 你 所 见 ， 目 动 生成 的 测试 用 例 失败 了 , 并 且 它 絮 叫 着 让 你 写 些 测试 用 例 。 那 就 写 吧 , 在 test 
文件 夹 里 写 个 core.clj 文 件 : 

这 个 测试 非常 简单 : 使 用 了 (deftest) 宏 ， 给 测试 命名 为 (one-day) ， 并 且 有 一 个 跟 断 言 
语句 非常 相似 的 形式 。Clojure 代 码 的 结构 使 得 (is) 形 式 读 起 来 非常 自然 一 一 几乎 就 像 DSL 一 样 。 
这 个 测试 可 以 读 作 “ 目 1970 年 1 月 2 日 以 来 的 毫秒 数 等 于 86 400 000 对 吗 ? ”我 们 来 看 一 下 这 个 测 
试 的 实际 效果 : 

Inas hello-lein.test,. core 


(:use [hello-lein.corel]) 
(uae [clojure.test])) 


Idefteast one-day 
(is tru 
(= Cd (igodate-to-millis-since-epoch "1970-01-02")))) 

这 里 的 关键 包 是 clojure.test, 它 提供 了 一 些 在 更 复杂 的 环境 或 需要 用 到 测试 固件 时 用 来 
构建 测试 用 例 的 形式 。 如 果 想 了 解 得 更 深入 , 请 参考 Amit Rathore 写 的 Clojure in Action ( Manning， 
2011 )， 其 中 对 Clojure 中 的 TDD 有 全 面 的 论述 。 

ariel:hello-=-lein boxcats lein test 

Testing hello-lein.test .core 


Ran 1 tests containing 1 assertions,. 
0 failures, 0 errors. 


在 面向 REPL 的 TDD 流 程 就 绪 后 ， 现 在 可 以 用 Clojure 构 建 一 个 具有 相当 规模 且 能 用 的 应 用 程 
序 了 。 但 你 终归 要 做 一 些 需 要 跟 人 分 享 的 东西 。 好 在 Leiningen 有 一 些 命令 可 以 让 你 很 容易 地 进行 
打包 和 部 署 。 


12.6.5 Leiningen 打包 和 部 署 

Leiningen 主 要 提供 了 两 种 代码 分 发 办 法 。 这 两 种 办 法 本 质 上 的 区 别 是 带 不 带 依 赖 项 。 对 应 的 
命令 分 别 是 lein jar 机 lein uberjar。 

我 们 先 看 一 下 lein jar: 


ariel:hello-lein boxcats lein jar 
Copying 4 files to /Users/boxcat/proljects/hello-lein/lib 
Created /Users/boxcat/projects/hello-lein/hello-lein-1.0.0-SNAPSHOT .jar 
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下 面 这 些 是 被 打包 进 JAR 文 件 中 的 东西 


ariel:hello-lein boxcats$ jar tvf hello-lein-1.0.0-SNAPSHOT .jar 
72 Sat Jul 16 13:38:00 BST 2011 META- INF/MANIFEST .MF 
1424 Sat Jul 16 13:39:00 BST 2011 META-INF/maven/hello-lein/hello-lein/ 
pom ,Xml 
105 Sat Jul 16 13:38:00 BST 2011 
META-INF/maven/hello-lein/hello-lein/pom.properties 
196 Fri Jul 15 21:52:12 BST 2011 proiject .cl1] 
238 Fri Jul 15 21:40:06 BST 2011 hello lein/core.cl] 
ariel:hello-lein boxcats 


其 中 最 明显 的 就 是 Leiningen 的 基本 命令 把 Clojure 源 文件 ， 而 不 是 编译 后 的 .class 文 件 发 出 去 
了 。 这 是 Lisp 代 码 的 传统 ， 因 为 系统 的 读 时 组 件 和 宏 会 因为 要 处 理 编译 后 的 代码 而 受到 阻碍 。 

现在 ， 我 们 来 看 看 用 lein uberjar 会 发 生 什么 。 它 所 产生 的 JAR 不 仅 包 含 代 码 ， 还 有 依 
赖 项 


ariel:;hello-lein boxcat$ lein uberjar 

cleaning up. 

Copying 4 fileas to /Users/boxcat/projects/hello-lein/lib 
Copying 4 files to /Userg/boxcat/projects/hello-lein/lib 
Created /Users,/boxcat/projects/hello-lein/hello-lein-1.0.0-SNAPSHOT . jar 
Including hello-=-lein-1.0.0-SNAPSHOT. jar 

Including cl]j]-time-0.3.0.J]ar 

Including clojure-1.2.1.Jar 

Including clolure=contrib-1.2.0.]ar 

Including joda-time-l1.6.jar 

Created /Users/boxcat/projects/hello-lein/ 

ms hello-lein-1.0.0-SNAPSHOT-standalone .Jar 


看 到 了 吧 ， 这 个 JAR 中 不 仅 有 代码 ， 还 有 依赖 项 ， 以 及 依赖 项 的 依赖 项 ， 这 称 为 依赖 的 传递 
闭 包 图 。 也 就 是 说 它 是 一 个 可 以 完全 独立 运行 的 包 。 

当然 ， 因 为 所 有 依赖 项 都 打包 了 ， 所 以 这 也 意味 着 lein uberjar 打 包 的 文件 要 比 lein jar 
的 文件 大 很 多 。 即 便 是 我 们 这 个 简单 的 小 例子 ， 其 差异 也 相当 鲜明. 


ariel:hello-lein boxcats$ ls -lh h* .jar 
-IW-I--r-- 1 boxcat staftf 4.1M 16 Jul 13:46 
hello-lein-1.0.0-SNMAPSHOT-=etandalone .Jar 
-TW=T==T== 1 boxcat staff 1.7K 16 Jul 13:46 
hello-lein-1.0.0-SNAPSHOT . jar 


你 可 以 这 样 理解 lein jar 和 lein uberjar: 如 果 要 构建 一 个 类 库 ( 构建 在 其 他 类 库 之 上 )， 
或 者 要 将 它 作 为 依赖 项 ， 就 用 1ein jar。 如 果 是 构建 一 个 最 终 用 户 使 用 的 Clojure 应 用 程序 ， 而 
不 是 交 给 用 户 去 扩展 的 工件 ， 就 用 lein uberjar。 

你 已 经 见 过 用 Leiningen 如 何 开 始 、 管理 、 构建 和 部 署 Clojure 项 目 了 。Leiningen 还 有 很 多 内 置 
的 实用 命令 , 还 有 一 个 强大 的 插件 系统 让 你 可 以 对 它 进行 定制 。 想 要 对 Leiningen 能 做 什么 有 更 多 
本 解 ， 只 要 调用 时 不 带 命 令 就 可 以 了 ， 单 用 Lein。 

我 们 在 下 一 章 构 建 Clojure 的 Web 应 用 时 还 会 过 到 Leiningen。 
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12.7 小结 


有 优秀 Java 开 发 人 员 参 与 的 项 目 肯 定 有 快速 、 可 重复 、 简 单 的 构建 。 如 果 软 件 不 能 快速 稳定 
地 构建 ， 就 会 浪费 大 量 的 时 间 和 金钱， 包括 你 自己 的 ! 

理解 编译 一 测试 一 打包 这 一 基本 构建 周期 是 确立 良好 构建 流程 的 关键 。 毕 葛 , 你 不 能 测试 还 
没 编译 的 代码 ! 

Maven 将 构建 周期 的 概念 发 扬 光 大 ， 扩 展 为 在 所 有 Maven 项 目 中 都 能 保持 一 致 的 项 目 周 期 。 
这 种 惯例 优先 ( 于 配置 ) 的 方式 对 于 大 型 软件 团队 非常 有 和 帮助， 但 有 些 项 目 可 能 需要 更 多 的 灵 
活性 。 

Maven 还 解决 了 依 颠 管理 的 问题 ， 因 为 几乎 所 有 项 目 都 要 依赖 第 三 方 类 库 ， 这 个 难题 
扰 着 Java 和 开源 世界 。 

把 构建 流程 挂 到 CI 环境 中 , 开发 人 员 就 能 得 到 迅捷 无 比 的 反馈 , 还 可 以 毫 无 娠 惧 地 快速 合并 
修改 。 

Jenkins 是 一 个 流行 的 CI 服务 兹 ,不仅 能 构建 几乎 所 有 类 型 的 项 目 , 还 能 通过 它 庞大 的 捅 件 系 
统 提供 丰富 的 报告 。 假 以 时 日 ， 开 发 团队 就 能 让 Jenkins 执 行 各 种 构建 ,覆盖 范围 可 以 从 快速 单元 
测试 构建 到 系统 集成 构建 。 

Leiningen 是 Clojure 项 目的 自然 之 选 。 它 用 一 个 非常 清爽 的 构建 和 部 署 工具 ， 把 紧凑 的 TDD 
循环 和 REPL 方式 结合 在 了 一 起 。 

我 们 接 下 来 会 讨论 快速 Web 开 发 ， 自 从 第 一 个 基于 Java 的 Web 框 架 出 现 以 来 ， 大 多 数 优 秀 的 
Java 开 发 人 员 都 曾 为 这 一 主题 奋斗 过 。 
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本 章 内 容 

日 为 什么 Java 不 是 快速 Web 开 发 的 理想 选择 
口 Web 框 加 的 选择 标准 

口 基于 JVM 的 Web 框 架 比 较 

日 认识 Grails ( 与 Groovy ) 

口 认识 Compojure ( 与 Clojure ) 


快速 Web 开 发 很 重要 ， 非 党 重要。 在 全 球 的 商业 和 社交 活动 中 ， 数 量 旋 大 的 网 站 和 由 Web 技 
术 驱 动 的 应 用 程序 占据 着 主导 地 位 。 企 业 ( 特别 是 创业 公司 ) 的 生死 取决 于 他 们 向 市 场 投放 新 产 
癌 或 新 特性 的 速度 。 如 今 的 终端 用 户 希 望 新 功能 的 出 现 和 bug 的 消失 能 像 变 鹿 术 一 样 快 ， 他 们 越 
来 越 没 耐心 了 。 

大 多 数 Java 上 的 Web 框 架 在 支持 快速 Web 开 发 的 能 力 上 都 有 限 ， 为 了 不 在 激烈 的 竞争 中 死 
控 ， 很 多 组 织 都 纷纷 转向 PHP 和 Rails 之 类 的 技术 。 

作为 一 个 优秀 的 Java 开 发 人 人员， 你 该 何去何从 ? 好 在 最 近 JVM 上 出 现 了 动态 层 语 言 ， 现 在 
JVM 上 也 有 快速 Web 开 发 的 理想 选择 。Grails ( Groovy ) 和 Compojure ( Clojure ) 就 是 这 样 的 框架 ， 
它们 能 满足 你 要 求 的 快速 Web 开 发 能 力 。 也 就 是 说 你 不 用 放弃 强大 而 又 灵活 的 JVM， 在 跟 PHP 和 
Rails 这 样 的 技术 竞争 时 也 不 用 比 它们 多 花 几 个 小 时 本。 


人 hil: Ne ah ei Ja a 中 1 ‘a EE 6 TT : Ja 快 I 速 We Yeb 开 发 是否 所 可 有 过 2 过 部 站 hf = J 
afp ( 兽 因 JSP、 Ba ti pe 天 og i EE ) 6 已 经 有 了 长 
的 发 展 。 尽 管 Java EE 6 所 做 的 改进 ( 在 JSP、Servlet 和 EJB API 上 有 明显 体现 ) 仍然 党 限于 Java 
静态 类 型 系统 和 编译 方面 的 问题 。 


本 章 一 开始 会 解释 一 下 为 什么 Java 上 的 Web 框 架 不 是 快速 Web 开 发 的 理想 选择 。 顺 着 这 个 解 
释 ， 你 会 了 解 优 秀 的 Web 框 架 应 该 满足 哪些 标准 。 通 过 一 些 定量 的 研究 ， 以 及 Matt Raible 的 工作 ， 
你 会 理解 如 何 用 Web 框 架 的 20 条 标准 对 各 种 JVM Web 框 架 进 行 评 级 。 

Grails 是 快速 Web 开 发 框架 的 领导 者 之 一 ， 它 满足 了 其 中 的 很 多 标准 。 我 们 会 带 你 过 一 遍 这 
个 基于 Groovy 的 Web 框 架 ， 科 手 可 热 的 Rails 框 架 对 它 产生 了 很 大 的 影响 。 
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我 们 还 会 讨论 作为 Grails 备 选 的 Compojure， 它 是 一 个 基于 Clojure 的 Web 框 架 ， 可 以 实现 非常 
精炼 的 Web 编 程 和 快速 开发 。 
让 我 们 先 来 看 看 为 什么 基于 Java 的 Web 框 架 不 一 定 是 现代 Web 项 目的 理想 选择 。 


13.1 Java Web 框架 的 问题 
第 7 章 讨 论 过 名 语 言 编 程 金字 塔 和 编程 语言 的 三 个 层次 ， 我 们 在 图 13-1 中 再 次 重复 一 -下 。 


图 13-1 多 语言 编程 金字 塔 


Java 位 于 稳定 层 , 所 以 它 的 所 有 Web 框 架 也 属于 这 一 层 。 作为 一 门 流行 而 又 成 识 的 语言 ，Java 
中 有 很 多 种 Web 框 架 ， 比 如 : 

DQ Spring MVC 

GWI 

DQ Struts 2 

D Wicket 

D lapestry 

口 JSF (及 其 他 与 Faces 相 关 的 类 库 ) 

DD Vaadin 

口 Play 

口 以 前 那些 普通 的 JSP/Servlet 

在 这 一 领域 ，Java 没 有 公认 的 领头 羊 ， 并 且 源 于 Java 的 这 一 分 支 根 本 就 不 是 快速 Web 开 发 的 
理想 选择 。Struts 2 曾经 流行 一 时 ， 该 项 目的 前 任 领导 者 说 过 这 样 一 段 话 : 


我 陡 落 了 :-), 我 现在 更 喜欢 用 Railks 一 一 因为 前 面 提 到 的 简洁 性 ,因为 我 再 也 不 用 “ 构 

建 ” 和 “部 署 ” 了 。 见 单 姐妹 们 ， 我 要 提醒 你 们 ， 如 果 你 们 想 吸 引 Rails 开 发 人 员 .…… 或 

者 要 避免 像 我 这 样 的 “ 背 薄 Java 的 Web 开 发 人 员 ” 流 失 :-)， 这 类 事情 是 你 们 必须 克服 的 
障 研 。 

一 一 Craig McClanahan，2007 年 10 月 23 日 

( http:Wmarkmail.org/thread/qfbSsekad33eobh2 ) 
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本 节 会 谈 到 Java 为 什么 不 是 快速 Web 开 发 的 理想 选择 。 我 们 先 来 看 看 编译 型 语言 为 什么 会 在 
开发 Web 应 用 时 拖 后 腿 。 


13.1.1 Java 编译 为 什么 不 好 


Java 是 编译 型 语言 ， 这 就 是 说 你 每 次 修改 代码 ， 都 必须 重复 下 而 的 步骤: 

口 重新 编译 Java 代 人 码 ; 

口 停止 Web 服 务 硕 ; 

口 把 改过 的 应 用 重新 部 署 到 Web 服 务 咒 中; 

口 启动 Web 服 务 器 。 

这 会 浪费 大 量 的 时 间 ! 特别 是 有 很 多 小 修改 时 ， 比 如 修改 控制 权 的 目标 或 界面 的 微调 。 


注意 ”应 用 服务 器 和 Web 服 务 器 之 间 的 界限 已 经 变 得 模糊 了 。 这 是 因为 JEE 6 的 出 现 ( 可 以 在 Web 
容器 内 运行 EJB )， 也 因为 大 多 数 应 用 服务 器 都 是 高 度 模 块 化 的 。 我 们 提 到 的 “Web 服 务 
器 ”是 指 任 何 有 Servlet 容 器 的 服务 器 。 


如 果 你 是 一 位 经 验 丰 请 的 Web 开 发 人 员 ， 应 该 知道 有 些 技术 可 以 解决 这 个 问题 。 其 中 大 多 数 
都 是 不 用 你 止 和 局 动 Web 服 务 硕 就 能 应 用 代码 修改 , 也 被 称 为 热 部 置 。 热 部 夏 或 者 把 全 部 资源 ( 比 
如 整个 WAR 文 件 ) 和 都 换 反 ， 或 者 只 选 其 中 几 个 【〈 比如 单个 JSP 页 面 ) 资源 进行 痊 换 。 但 热 部 普 不 
是 100% 可 靠 ( 因为 类 加 载 的 限制 和 容器 中 的 bug )， 并 且 大 多 数 情况 下 ，Web 服 务 器 仍然 需要 执行 
昂贵 的 代码 重 编译 操作 。 


pr rr | rp py ( aa 


eroturnaround.com/jrebel/ )。 JRebel 确 实 具备 一 些 惊艳 的 JVM 技 巧 , 它 处 在 IDE 和 Web 服 务 器 
之 间 ， 源 码 发 生变 化 时 能 自动 将 这 些 变化 反应 到 正在 运行 的 Web 服 务 器 上 (LiveRebel 用 于 生 
产 环境 的 部 署 )。 这 种 热 部 署 基本 没什么 问题 ， 并 且 这 些 工 具 实际 上 是 解决 热 部 署 问题 的 行 
业 标 准 。 


一 般 来 说 ，Java Web 框 法 为 代码 修改 而 产生 的 周转 时 间 太 长 。 但 这 不 是 唯一 的 问题 ， 另 外 一 
个 拖 慢 Web 开 发 速度 的 不 利 因素 是 语言 的 灵活 性 ， 这 是 静态 类 型 系统 的 弱项 。 
13.1.2 静态 类 型 为 什么 不 好 

在 开发 新 产品 或 新 功能 的 早期 阶段 , 保持 用 户 展示 屋 设 计 的 开放 性 ( 对 于 类 型 而 言 ) 通常 都 


是 明智 之 举 。 对 于 用 户 来 说 ， 要 求 数值 精确 到 小 数位 ,或 让 书 单 变 成 图 书 和 玩具 混合 的 清单 都 是 
非常 合理 的 要 求 。 静 态 类 型 可 能 会 成 为 这 类 需求 的 巨大 障碍 。 如 果 你 必须 把 一 个 Book 对 象 列表 
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变 成 BookorTov 对 和 象 的 列表 ， 则 只 能 在 代码 中 把 这 个 静态 类 型 全 改 掉 。 

尽管 在 容器 类 里 能 用 基本 类 型 ( 比如 Java 的 object 类 ) 作为 对 象 的 类 型 ,但 这 肯定 不 是 最 
佳 实践 一 一 这 何 直 是 一 下 于 回 到 了 这 型 。 

因此 ， 选 择 基于 动态 层 语言 编写 的 Web 框 架 无 疑 是 个 有 效 选 项 。 


注意 ”Secala 当 然 是 静 坊 类 型 语言 。 但 由 于 其 先进 的 类 型 推断 能 力 ， 它 能 规避 很 多 由 Java 静 坊 类 
型 实现 方式 引发 的 问题 。 也 就 是 说 ，Seala 可 以 是 、 也 确实 是 可 用 的 Web 层 语言 。 


在 你 继续 深入 人 研究 和 选择 Web 开 发 的 动态 语言 之 前 ,我们 先后 进一步， 让 视野 更 开阔 一 点 。 
想 一 想 优秀 的 快速 Web 开 发 框架 应 该 符合 哪些 标准 。 


13.2 ”选择 Web 框架 的 标准 


Java 这 么 多 年 的 顶级 编程 砍 言 地 位 不 是 日 给 的 ，Java 中 可 供 选 择 的 Web 框 架 有 很 多 。 最 近 基 
于 其 他 JVM 语 言 ( 比如 Groovy、Scala 和 Clojure ) 的 Web 框 架 也 雄 起 了 。 但 不 幸 的 是 ,这么 多 年 来 
还 没有 一 个 能 在 这 一 领域 确立 自己 的 霸主 地 位 ， 选 哪个 框架 完全 看 个 人 偏好 。 

Web 框 架 应 该 能 提供 大量 帮助 。 必 须 能 用 一 些 标准 进行 评估 ， 它 能 满足 的 标准 越 多 ， 开 发 
Web 应 用 的 速度 可 能 会 越 快 。 

Matt Raible* 总 结 出 了 评判 Web 框 架 的 20 条 标准 ”。 表 13-1 对 这 些 标准 作 了 简要 介绍 。 


表 13-1 Web 框 架 的 20 条 标准 


标 准 示 全 
开发 人 员 的 工作 效率 能 用 1 天 或 5 天 搭 出 一 个 CRUD 页 面 吗 
开发 人 员 的 看 法 用 起 来 有 意思 吗 
学 习 曲 线 学 了 一 个 礼拜 或 一 个 月 后 能 干 活 吗 
项 目 健康 状况 项 目 陷 人 绝境 了 吗 
开发 人 员 的 充足 性 能 找到 经 验 丰富 的 开发 人 员 吗 
就 业 趋 妇 将 来 能 招 到 人 吗 
模板 化 遵循 DRY ( 不 重复 自己 ) 原则 吗 
组 件 自 带 日 期 选择 器 之 类 的 东西 吗 


中 小 心 ! 如 果 你 的 域 对 象 中 有 Dr 或 And 这 样 的 字眼 出 现 ， 那 你 很 可 能 违反 了 我 们 在 第 11 章 讨论 的 SOLID 原 则 。 

四 AppFuse ( httpJ/appfuse.org ) 的 作者 ，AppFuse 是 一 个 Web 开 发 基础 平台 ， 它 储 成 了 Java 中 各 种 流行 的 Web 框 架 ， 
并 提供 了 所 有 Web 系 统 开 发 过 程 中 都 需要 开发 的 一 些 功能 .比如 登录 、 用 户 密码 加 密 、 用 户 管理 、 为 不 同 的 用 户 
展现 不 同 的 药 单 。 它 可 以 自动 生成 40%-60% 的 代码 ， 还 自 带 了 一些 默认 的 CSS 样 式 ， 使 用 这 些 样 式 能 快速 改变 整 
个 系统 的 外 观 ， 还 具备 自动 测试 的 能 力 。 纤 者 注 

Matt Raible，“Comparing JVM Web Frameworks” (March 2011 ) , presentation. http://raibledesigns.com/rd/page/ 
publications . 
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( 续 ，) 


标准 示例 
Ajax 支持 客户 端的 异步 Javascript 调 用 吗 
插件 或 附加 项 能 加 上 Facebook 集 成 之 类 的 功能 吗 
扩展 性 默认 的 控制 器 处 理 的 并 发 用 户 数 能 到 500+ 吗 
测试 支持 能 做 测试 驱动 的 开发 吗 
I1g8n 和 110n 自 带 其 他 语种 和 地 域 的 支持 吗 
校 验 能 轻松 校 验 用 户 输入 并 迅速 反馈 吗 
多 看 盲 豆 皖 能 同时 用 (比如 说 )Java 和 Groovy 吗 
文档 /教程 的 质量 常见 的 用 例 和 问题 在 文档 中 都 有 体现 吗 
出 版 图 书 有 没有 行业 专家 用 过 它 ， 并 分 享 了 自己 的 战斗 事迹 
REST 支 持 ( 服务 器 端 和 客户 端 ) 它 能 按 HTTP 的 设计 宗旨 使 用 该 协议 到 
移动 艾 持 是 否 很 容易 就 能 支持 Android、iDS 和 其 他 移动 设备 
风险 程度 是 用 来 做 “保存 食谱 ”的 应 用 程序 或 是 “ 校 电站 控制 器 "” 


你 看 到 了 ， 这 个 清单 很 长 ， 在 做 决定 时 ， 你 需要 想 好 各 个 标准 的 权重 。 不 过 Matt 很 勇敢 ?， 
他 最 近 在 这 一 领域 做 了 一 些 研究 , 尽管 其 研究 结果 引发 了 激烈 的 争论 , 但 真相 总 算是 开始 浮现 了 。 
如 果 给 那些 对 快速 Web 开 发 最 重要 的 标准 赋予 较 高 的 权重 ， 各 种 框架 的 得 分 ( 总 分 为 100 ) 如 图 
13-2 所 示 。 这 些 标准 应 该 是 ， 开 发 人 员 的 工作 效率 、 测 试 支持 和 文档 的 质量 。 


图 13-2 Matt Raible 对 JVM 框 音 的 加 权 评 级 


不 同 的 人 需求 可 能 会 不 同 ， 在 http:Wbitlyjvm-frameworks-matrix 上 可 以 很 容易 地 修改 Matt 的 
权重 ， 运 行 目 己 的 分 析 ， 产 生 目 已 的 图 形 。 


加 你 应 该 能 想象 得 到 ， 人 们 对 于 自己 喜爱 的 Web 框 架 有 多 大 的 热情 ! 
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提示 “在 你 选 定 框架 之 前 ， 我 们 强烈 建议 你 在 两 到 三 个 框架 上 按 自己 的 标准 做 一 些 功能 原型 。 


现在 你 知道 该 用 哪些 标准 进行 评估 了 ， 并 且 还 能 利用 Matt 提 供 的 工具 ， 所 以 在 选择 快速 Web 
开发 框架 时 ， 你 可 以 做 出 明智 的 选择 。 在 我 们 的 加 权 标 准 分 析 中 脱颖而出 的 是 Grails 框 架 
(Compoijure 设 有 名 列 前 切 , 但 因为 它 还 非常 年 轻 ， 所 以 我 们 预计 它 在 不 久 的 将 来 能 迅速 蹄 升 到 领 
导 阵 营 中 )。 

我 们 来 看 看 获胜 者 Grails! 


13.3 Grails 入 门 


Grails 是 基于 Groovy 的 快速 Web 应 用 框架 , 它 集 成 了 多 个 第 三 方 类 库 , 包括 Spring、Hibernate、 
JUnit 和 Tomcat 服 务 估 等。 它 是 一 个 完备 的 Web 框 架 ，13.2 节 列 出 的 20 条 标准 它 全 都 满足 。 还 有 一 
点 很 重要 ，Grails 在 很 大 程度 上 借鉴 了 Rails 中 惯例 优先 的 原则 。 如 果 能 依照 惯例 编码 ， 框 架 会 帮 
你 做 很 多 套路 化 的 工作 。 

我 们 在 这 一 节 中 会 讨论 如 何 措 建 你 的 第 一 个 快速 启动 应 用 。 在 搭建 快速 启动 应 用 的 过 程 中 ， 
你 会 看 到 很 多 可 以 证 明 Grails“ 快 速 ” 的 证 据 。 我 们 还 会 指出 Grails 中 那些 需要 进一步 探索 的 重要 
技术 ， 从 而 让 你 可 以 构建 出 能 用 于 生产 环境 的 、 正 儿 八 经 的 应 用 程序 。 
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es ype a 我 们 觉得 它 没 Grails 成 熟 ， 
但 如 果 你 确实 不 喜欢 Groovy， 这 也 是 个 不 错 的 备 选 。 


如 果 不 熟 悉 Groovy, 可 能 需要 认真 温习 一 下 第 8 章 , 在 你 能 跟 Groovy 融 治 相 处 后 ,请 下 载 Grails 
并 安装 。 附 录 C 中 有 该 过 程 的 完整 指导 。 
装 好 Grails 之 后 ， 就 该 开始 你 的 第 一 个 Grails 项 目 了 ! 


13.4 ”Grails 快速 启动 项 目 


这 一 帮会 介绍 一 个 Grails 快 速 司 动 项 目 ， 重 点 展示 Grails 作 为 快速 Web 框 架 的 亮点 。 用 Grails 
创建 Web 应 用 所 需 的 步骤 如 下 : 

口 创建 域 对 象 ; 

口 测试 驱动 开发 ; 

口 域 对 象 的 持久 化 ; 

口 创建 测试 数据 ; 


1 地 
OEEEE 
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口 GSP 视 图 ; 

口 脚手架 和 自动 化 的 UI 创 建 ; 

口 快速 开发 的 周转 时 间 。 

说 得 具体 点 ， 我 们 准备 摘 一 个 角色 扮演 游戏 "中 的 基本 构件 ( Playercharacter )。 到 本 节 
结束 的 时 候 ， 你 会 创建 一 个 具备 以 下 能 力 的 简单 的 域 对 象 ( Playercharacter ): 

口 进行 一 些 运行 时 测试 ; 

口 预先 准备 好 测试 数据 ; 

口 可 以 保存 到 数据 库 中 ; 

口 具有 可 以 进行 CRUD 操 作 的 基本 UI。 

Grails 六 省 时 间 的 第 一 个 法 宝 就 是 自动 创建 好 项 目 结构 。 运 行 grails create-app 
<my-project> 命 全， 马上 就 能 得 到 一 个 可 以 构建 的 项 目 ! 你 需要 做 的 唯一 一 件 事情 就 是 保证 能 接 
人 互联 网 ， 因 为 它 要 下 载 标准 的 Grails 依 赖 项 ( 比如 Spring、Hibemate、JUnit、Tomceat 服 务 器 等 )。 

Grails 用 来 管理 和 下 载 依 赖 项 的 工具 是 Apache Ivy。 它 下 载 和 管理 依赖 项 的 概念 跟 第 12 章 介绍 
的 Maven 非 常 像 。 下 面 这 个 命令 会 创建 一 个 叫做 pcgen_grails 的 应 用 程序 ， 包 括 一 个 依照 Grails 的 
传统 优化 过 的 项 目 结构 。 

grails create-app pcgen grailas 

依 辆 项 下 载 完成 ， 其 他 目 动 安 沪 步 又 也 完成 之 后 ， 你 应 该 就 会 得 到 一 个 如 图 13-3 所 示 的 项 目 


application.properties -> basic application info/versioning 


+ grails-app 
+ conNf 一 > location of configuration artifacts 
+ hibernate 一 > Optional hibernate configuration 
+ Spring 一 > optional spring i 
+ controllers —> 1 of controller 
+ Gomain 一 > location of domain classes 
+ i1Bn —> location of message bundles for i18n 
+ SBrvices —> location of services 
+ taglib 一 > location of tag libraries 
+ ViEWS 一 > location of views 
+ tayouts 一 > |ocation of layouts 
+ lib 
+ scripts 一 > Scripts 
+ Src 
+ groovy 一 > Oplional location for Groovy source flles 
(of types other than those in grails-app/”) 
+ java -一 > DOplional location for Java source files 
+ test 一 > generated test classes 
二 
+ WEB-INF 


图 13-3 ”Grails 项 目的 布局 
有 了 项 目 结构 就 可 以 开始 生产 一 些 能 运行 的 代码 了 ! 首先 要 创建 域 对 象 类 。 


山 想 想 《 龙 与 地 下 城 》 或 《指环 王 》 


1 入 
三 
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13.4.1 创建 域 对 象 


Grails 以 域 对 象 为 应 用 程序 的 核心 , 因此 辟 励 你 按 域 驱动 设计 ( Domain-Driven Design, DDD ) 
的 方式 来 考虑 问题 "。 执 行 grails create-domain-celass 命 邻 可 以 创建 域 对 象 。 
下 面 的 例子 创建 了 一 个 Playercharacter 类 ， 用 来 表示 游戏 中 的 角色 : 


cd pcgen grails 
grails create-domaln-class com.Jjava7developer.chapterl3.PlayerCcharacter 


Grails 会 目 动 为 你 创建 下 面 的 文件 ; 
口 一 个 表示 域 对 象 的 PlayerCharacter.groovy 源 文件 ( 在 目录 grails-app 
chapterl3 下 ); 
口 开发 单元 测试 用 的 PlayerCharacterTests.groovy 源 文件 ( 在 目录 test/unit/com/java7developer/ 
chapterl3 下 )。 
看 ，Grails 在 辟 励 你 写 单 元 测试 ! 
还 需要 给 PlayerCharacter 定 义 一 些 属性 ， 比 如 strength、dexterity 和 charisma。 有 
了 这 些 属 性 ， 你 就 可 以 开始 构想 游戏 中 的 角色 如 何 跟 想象 的 世界 交互 *。 但 刚刚 看 过 第 11 章 ， 你 
当然 想 先 写 测试 ! 


13.4.2 ”测试 驱动 开发 


按 TDD 的 方式 ， 我 们 要 先 写 个 失败 测试 ， 然 后 实现 Playercharactezr 让 测试 通过 。 

我 们 还 准备 利用 Grails 的 域 对 象 自动 校 验 特性 。 在 Grails 中 ， 可 以 自动 在 任何 域 对 象 上 调用 
validate() 方 法 ， 以 确保 该 对 象 的 有 效 性 。 代 码 清单 13-1 会 测试 strength、dexterity 和 
charisma 三 项 统计 量 都 是 3 到 18 之 间 的 数值 。 


lomain/com/java7developer/ 


package com.JavaTdeveloper. se 


import grails.test.* 
? 3 扩展 Grails Unit- 
Class PlayerCharacterTests extends GrailstUnitTestCagse | TestCase 
PlayerCharacter pe; 
protected void setUp() 1{ 
super .setUpl) 0 注入 validata() 
mockForConstraintsTests (PlayerCcharacter) 


| 
protected void tearDown() 1 
SUper .tearDown') 


| 


山 想 了 解 DDD ( 由 Eric Evans 提 出 ) 的 更 多 内 容 ， 请 访问 域 驱动 设计 社区 http://domaindrivendesign_org/ )。 
四 Gweneth 是 不 是 应 该 善于 摔跤 、 杂 机 ,或 面 带 微笑 地 解除 对 手 的 武装 ? 


1 入 
三 
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void testConstructorSucceedsWithValidAttributes 1 : 
pe = new PlayerCcharacter(3, 535, 18) 3) 通过 校 验 
assgert pce .validatel) 3 


void testConstructorFailsWithSsomeBadAttributes() | 


pe = new PlayercCharacter (10, 19, 21) 从 校 验 失败 
assertFalse pe.validate!) < 


} 

} 

Grails 的 单元 测试 都 应 该 扩展 自 GrailsUnitTestcase 因 。 跟 所 有 标准 的 JUnit 测 斌 一样， 它 
也 有 setUp() 和 tearDown () 方 法 。 但 为 了 在 单元 测试 阶段 用 Grails 内 置 的 validate() 方 法 ， 必 
须 通 过 mockForConstraintsTest 方 法 把 它 拉 进 来 傣 。 这 是 因为 Grails 把 valiqate () 当做 集成 
测试 的 关注 点 ,通常 只 有 这 样 才 能 用 它 。 但 如 果 想 要 更 快 地 得 到 反馈 ， 可 以 把 它 放 到 单元 测试 阶 
段 。 接 下 来 ， 可 以 调用 valiaate() 来 检查 域 对 象 是 否 有 效 @@。 

现在 可 以 执行 下 面 的 命令 来 运行 测试 了 ， 

grails test-app 

这 个 命令 既 运 行 单元 测试 也 会 运行 集成 六 
中 的 输出 来 看 ， 测 试 失败 耳 。 

要 了 解 测 试 失败 的 原因 ， 需 要 到 target/test-reports/plain 目 录 下 去 找 。 对 于 这 个 程序 ， 要 找到 
TESTunit-unit-com,java7developerchapter13.PlayerCharacterTests.txt 文 件 。 这 个 文件 会 告诉 你 测试 
失败 是 因为 在 尝试 创建 新 的 Playercharacter 时 ， 没 找到 匹配 的 构造 方法 。 这 很 容易 理解 ， 因 
为 Playercharacter 域 对 象 还 什么 都 没有 呢 ! 

现在 你 可 以 把 Playercharacter 措 起 来 , 重复 运行 测试 直到 通过 。, 按 你 的 想法 加 上 strength.、 
aexteritv 和 charisma 三 个 属性 。 但 为 了 在 这 些 属 性 上 设 定 minimum (3) 和 maximum(18) 的 限 
制 ， 需 要 用 特殊 的 限定 语法 。 那 样 就 可 以 用 Grails 提 供 的 默认 valiaate() 方 法 了 。 


则 试 〈 不 过 我 们 现在 只 有 单元 测试 )， 并 且 从 控制 台 


需求 。 ra 了 (和 人 2 中 朋 Tainfmax Eeeporeaee 
限定 。 参见 http://grails.org/doc/latest/guide/validation.h / 


下 面 这 段 代码 中 的 playercharacter 类 中 仅 包含 了 让 它 可 以 通过 测试 的 最 基本 的 属性 和 限定 。 
代码 清单 13-2 playerCharacter 类 


package com. ava7rdeveloper.chapterl3 
class Playercharacter | 


Integqer strength 
Integer dexterity 月 要 持久 化 的 类 型 变量 


Integqer charisma 


FlayerCcharacter() 1| 


1 二 
OEEEE 
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PlayYerCharacter(InEegqer str, Integer dex，Integer cha) 1 
strength = str 
dexterity = dex 
charisma = cha 
} 
static constraints = { 
strength(min:3, max:18) 
dexterity (min:3, max:18) 
charisma (min:3, max:18) 


可 以 通过 测试 的 
构造 方法 


用 于 校 验 的 限定 


} 

Playercharacter 类 相当 简单 。 有 三 个 会 自动 保存 到 Playercharacter 表 中 的 基本 属性 倍 。 
有 一 个 带 dig 的 构造 方法 合 。 那 个 特殊 的 static 代 码 块 确定 了 valiqate() 方 法 要 检查 的 

maxll 

oo 盾 古 半生 和 测试 应 该 能 很 痛快 地 通过 了 (再 次 运行 grails 
cest-app )。 如 果 遵 循 TDD 方 式 ， 到 这 个 阶段 就 该 着 手 重 构 Playercharacter 和 测试 了 ， 以 便 
让 代码 更 加 清爽 。 

Grails 还 会 确保 域 对 象 保存 到 数据 存储 中 。 
域 对 象 持久 化 

持 从 化 是 由 Grails 上 自动 处 理 的 , 因为 Grails 认 为 类 中 所 有 具有 明确 类 型 的 域 变量 都 应 该 保存 到 
数据 库 中 。Grails 会 自动 把 域 对 象 映射 到 同名 的 表 中 。 对 于 playercharacter 域 对 象 而 言 ， 三 个 
属性 ( strength 、dexterity 和 charisma ) 全 部 是 Integer 类 型 ， 所 以 都 会 映射 到 
PlayerCharacter 表 中 。Grails 默 认 使 用 Hibemate， 并 会 提供 一 个 HSQLDB 内 存 数 据 库 (我们 在 
第 11 章 提 到 过 它 ， 那 时 用 做 伪装 测试 替身 )， 但 你 可 以 用 自己 的 数据 源 取代 默认 数据 源 。 

grails-app/conf/DataSource.groovy 文 件 里 是 数据 源 的 配置 。 可 以 在 这 里 为 每 种 环境 设 定数 据 
源 。 记 住 ，Grails 已 经 在 pcgen_grails 里 给 出 了 默认 使 用 HSQLDB 的 实现 ， 所 以 无 需 任 何 修改 就 可 
以 运行 它 。 但 代码 清单 13-3 中 给 出 了 作息 玫 所 库 的 本 置 供 参 照 。 
代码 清单 13-3 ”可 能 的 pcg 


dataSource || 


environments 1 
development {| dataSource {} |} 


test {| dataSource {|} | 
生产 数据 源 
production | z i 


dataScource | 


dbCreate = " update 数据 和 i 天 | 
| | | 数 动 
driverClassName = "com.myasgql .jabc.Drivern 库 驱 


url = "jdbcec:mysql://localhoat /my app" 
USEername = "FOOEt" = 
二 JDBC 连 接 URL 


password = 


1 入 
Yr 
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比如 说 ， 可 以 在 生产 环境 中 使 用 MySQL 数 据 库 ， 而 开发 和 测试 环境 中 还 用 HSQLDB。 这 些 
都 是 相当 标准 的 Java 数 据 库 连 接 (JDBC ) 配置 ， 你 对 它们 应 该 已 经 很 熟悉 了 。 

Grails 开 发 者 也 考虑 到 了 手工 创建 测试 数据 的 问题 ， 所 以 他 们 提供 了 一 种 机 制 ， 可 以 在 应 用 
启动 时 将 数据 预 填充 到 数据 库 中 。 


13.4.4 创建 测试 数据 


测试 数据 的 创建 通常 是 由 Grails 的 BootStrap 类 完成 的 , 它 在 grails-app/confBootStrap.groovy 
中 。 只 要 Grails 应 用 或 Servlet 容 髓 启动 ， 就 会 运行 init 方 法 。 这 和 大 多 数 Java Web 框 架 用 的 启动 
servlet 所 起 的 作用 是 一 样 的 。 


注意 ”可 以 用 Bootstrap 类 做 所 有 初 地 化 操作 ， 和 es 主要 讨论 斌 数据。 


代码 清单 13-4 在 初始 化 阶段 生成 了 两 个 Playercharacter 域 对 象 , 并 把 它们 存 到 了 数据 库 里 。 
代码 清单 13-4 ”为 pcgen_grails 引 导 测 试 数据 


import com.Java7developer.chapterl3.PlayerCharacter 


class BootStrap | 有 在 Servlet 上 下 文 
启动 时 引导 


def init = | servletContext -> 村 
if {lpPlayerCharacter.count(})) 1 
new PlayerCharacter (strength: 3, dexterity: 5, charisma: 18) 
= .Save lfailoOnError: true) 
new PlayerCcharacter lstrength: 18, dexterity: 10, charisma: 4) 
mi .SAVEe (failonError: true) 
| 
| 
def destroy = {} 
| 
每 次 把 代码 部 署 到 Serviet 容 器 中 都 会 执行 init 方 法 ( 即 应 用 启动 和 Grails 自 动 部 署 时 ) @，。 
为 了 确保 不 会 禾 盖 掉 任 何 已 有 数据 ， 可 以 对 已 有 的 Playercharacter 实 例 执行 简单 的 count () 
方法 。 如 来 确定 没有 实例 ， 可 以 创建 一 些 。 这 里 有 个 很 重要 的 特性 如 果 有 异常 抛 出 ,或 所 构造 
的 对 象 无 法 通过 校 验 , 则 可 以 肯定 对 象 不 会 保存 到 数据 库 中 。 如 果 愿 意 , 可 以 在 aestroy 方 法 中 
执行 清除 操作 。 
有 了 一 个 带 有 存储 支持 的 基本 域 对 象 后 就 可 以 进入 下 一 阶段 了 : 在 Web 页 面 上 显示 域 对 象 。 
为 此 需要 构建 一 个 Grails 控 制 器 ， 你 应 该 不 会 对 这 个 源 自 MVC 设 计 模式 的 术语 感到 陌生 。 


13.4.5 ”控制 器 
Grails 钵 循 MVC 设 计 模 式 ， 用 控制 纶 来 处 理 来 自 客户 端 (一 般 是 浏览 器 ) 的 Web 请 求 。Grails 
的 惯例 是 给 每 个 域 对 象 配 一 个 控制 器 。 


3 者 
信和 目 在 读书 @3 
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要 创建 域 对 象 PlayerCcharacter 的 控制 器 只 需要 执行 下 面 这 条 命令 : 
gqrails create-controller com.JjavaTdeveloper.chapterl3.PlayerCharacter 
重要 的 是 指明 域 对 象 的 完全 限定 类 名 ， 包 括 包 名。 
命令 执行 完成 后 应 该 能 发 现下 面 这 些 文件 : 
口 Playercharacter 域 对 象 的 控制 器 的 PlayerCharacterController.groovy 源 文件 (在 
grails-app/controller/com/java7developer/chapter13 目 录 下 ); 
口 开 发 控制 器 单元 测试 的 PlayerCharacterControllerTests.groovy 源 文件 (在 testunitcomy 
java7developer'chapter13 目 录 下 ); 
口 grails-app/view/playerCharacter 文 件 夹 ( 稍 后 会 用 到 )。 
控制 器 以 简单 的 方式 支持 REST 风 格 的 URL 和 操作 映射 。 假 设 要 把 REST 风 格 的 URL 
http://localhost:8080/pcgen grails/playerCharacter/list 映 射 到 一 个 返回 Playercharacter 对 和 象 列表 
的 方法 上 。 按 照 Grails 惯 例 优 于 传统 的 方式 可 以 用 最 少 的 源码 把 URL 映 射 到 
PlayerCharacterController 类 中 。 这 个 URL 是 由 下 面 这 些 元 素 组 成 的 ， 
口 服务 器 ( http://localhost:8080/ ); 
口 基础 项 目 ( pcgen grails/ ); 
口 控制 入 名 称 的 衍生 部 分 ( playerCharacter/ ); 
在 控制 如 里 声明 的 操作 块 变量 ( list )。 
要 在 代码 中 看 到 这 些 元 率 ， 请 用 代码 清单 13-5 替 换 E 


代码 清单 13-5 playercharactercontroller 
package com.jJavaTdeveloper.chapter13 


class PlayerCharacterController | 
Ligst playerCharacters 


def ligst = 1 9 温 回 PlayerCcharacter 
playerCcharacters = PlayerCcharacter.liest|() 对 莹 列表 


| 

使 用 Grails 的 惯例 处 理 方式 ，plavercharactezr 的 属性 会 用 在 REST 风 格 的 URL 指 向 的 页 面 
+Q. 

但 如 果 现 在 就 启动 程序 ， 然 后 访问 http://localhost:8080/pcgen grails/playerCharacter/list， 是 不 
会 成 功 的 ， 因 为 还 没 创建 JSP 或 GSP 页 面 。 现 在 我 们 就 来 解决 这 个 问题 。 


13.4.6 GSP/JSP 页 面 


用 Grails 既 可 以 创建 GSP 页 面 ， 也 可 以 创建 JSP 页 面 。 这 一 节 会 创建 一 个 简单 的 GSP 页 面 ， 用 
来 显示 Playercharacter 对 象 的 列表 (设计 师 、Web 开 发 者 和 HTML/CSS 大 合 们 ， 现 在 请 移 开 
你 们 的 视线 ! ) 

代码 清单 13-6 是 GSP 页 面 grails-app/view/playerCharacter/list.gsp 的 代码 。 


由 自在 读书 他 


WWW.ZiZzidiary.com 
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<body> 
<hli>Po'sec /hls 
<tables 
<thead> 


| 
<tdsStrength< /td> 
<td>Dexterity</tds 
<tdscharisma</tds 
/tr> 
</theads> 


<tbody> 开始 循环 
< 多 plaverCharacters.each({ pe -> 和 > 


tr» 
tdsc$="$Ipc? .astraength}"$Yse/tds | \ ,2 
去 蕊 全 > 过量 三 1 六 (PIC3 .dexterity} "></tds 已 输出 属性 
< 七 品 ><< 生 二 "和 (1 PC? .chariema} "b></tds 

/tr 

< 车 | ) 告 > _ | 

/theads 和 
</table> 0 循环 结束 
</body> 
/htmls 


HTML 非 常 简单 ， 关 键 是 如 何 用 Groovy 肢 本。 你 会 注意 到 我 们 在 第 8 章 介 绍 的 Groovy 函 数字 
面值 语法 ， 它 简化 了 集合 循环 操作 略 。 接 着 是 对 角色 属性 的 引用 ( 注意 安全 的 null 解 引用 操作 
符 的 使 用 ) 人 @， 然 后 结束 函数 字面 值 舍 。 

既然 域 对 象 、 控 制 器 和 它 的 显示 页 面 都 准备 好 了 ， 接 下 来 就 可 以 局 动 Grails 应 用 了 ! 执行 下 
而 这 条 命令 即 可 : 

grails run-app 


Grails 会 自动 在 http://localhost:8080 上 启动 一 个 Tomcat， 并 把 pcgen grails 应 用 部 署 上 去 。 


警告 很 多 开发 人 员 已 经 装 过 Tomcat 服 务 器 了 ,如果 想 同 时 启动 多 个 Tomcat 实 例 ， 就 要 修改 端 
口号 ， 端 口 89080 只 能 有 一 个 实例 监听 。 


如 果 你 打开 浏览 髓 访问 http:Wlocalhost:8080/pcgen grails/ ， 会 看 到 页 面 上 列 出 了 
PlayerCharacterController, 如 1 图]13-4 有 TA 未 。 

点 击 com.java7developer.chapter13.PlayerCharacterController 链 接 ， 就 会 进 人 PlayerCharacter 域 
对 象 的 列表 页 。 

尽管 做 这 个 GSP 页 面相 当 快 ， 但 如 果 框 架 能 帮 你 做 岂 不 是 更 好 ? 用 Grails 的 脚手架 功能 可 以 
讯 速 做 出 域 对 象 CRUD 页 面 的 原型 。 
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图 13-4 pcgen grails 主 页 


13.4.7 ”脚手架 和 Ul 的 自动 化 创建 


Grails 吕 以 用 它 的 脚手架 ( scaffolding ) 特性 自动 创建 用 来 执行 域 对 象 CRUD 操 作 的 UL。 
要 使 用 脚手架 特性 , 请 用 代码 清单 13-7 赫 换 PlayerCharacterController groovy 源 文件 中 的 代码 : 


5 月 军 15- 市 有 赔 才 堵 playerCharacterController 
PR com. javaTdeveloper.chapterl3 


et Fe 

PlayerCharacterController 类 非常 简单 。 依 照 懈 例 将 域 对 象 的 名 称 赋值 给 脚手架 变量 
合 ，Grails 马 上 就 可 以 构建 默认 UI。 

请 暂时 把 1ist .gsp 改 成 list_original .gsp， 以 防 它 会 妨碍 脚手架 产生 相应 的 文件 。 改 
好 之 后 ， 刷 新 http://localhost:8080/pcgen grails/playerCharacter/list 页 面 ， 就 会 看 到 自动 生成 的 
PlayerCharacter 域 对 和 象 列 表 ， 如 图 13-$ 所 示 。 


Hi hs PT 


PileveC hracta Ls 


a 1 由 


图 1]13-5 PlayerCcharacter 实 例 列 表 
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在 这 个 页 面 中 也 可 以 创建 、 更 新 和 删除 Playercharacter 对 象 。 请 确保 添加 了 两 个 
PlayerCharacter 域 对 象 记录 ， 然 后 进入 下 一 节 了 解 与 代码 修改 的 快速 周转 有 关 的 内 容 。 
13.4.8 快速 周转 的 开发 

Grails 的 run-app 命 令 为 快速 Web 开 发 中 的 “快速 ”页 献 了 一 点 儿 特 殊 的 东西 。 用 Grails 的 


run-app 命 令 运行 的 应 用 程序 , 其 源码 会 和 服务 器 连接 起 来 。 尽管 这 在 生产 环境 中 不 是 什么 明智 
之 举 ( 因为 会 影响 性 能 )， 但 对 于 开发 和 测试 来 说 非常 重要 。 


提示 ”对 于 生产 环境 ， 一 般 都 是 用 grails war 创 建 WAR 文 件 ， 然 后 通过 标准 的 开发 流程 进行 
部 省 。 


如 果 Grails 应 用 中 的 源码 改 了 ， 这 些 变化 会 自动 反映 到 服务 器 上 ”。 我 们 来 试 试 ， 改 一 下 
PlayerCharacter 域 对 象 : 在 PlayerCharacter.groovy 文 件 中 加 一 个 变量 name， 存 一 下 。 

string name = 'Gweneth the Mercileses' 

现在 刷新 http://localhost:8080/pcgen grails/playerCharacter/list 页 面 ， 就 能 看 到 Playercharacter 
对 象 上 新 加 了 name 属 性 这 一 列 。 注 意 到 了 吗 ? 不 用 停 Tomcat， 不 用 重新 编译 代码 ， 其 他 的 什么 也 
不 用 做 。Grails 就 是 罪 这 种 几乎 即时 生 歼 的 速度 确立 了 它 快速 Web 开 发 框 滁 的 领导 地 位 。 

我 们 对 快速 启动 现 目 的 介绍 就 到 此 为 止 了 ， 你 应 该 体验 了 一 把 用 Grails 做 快速 Web 开 发 。 当 
然 ， 还 有 很 多 可 以 对 默认 行为 进行 定制 的 方法 值得 探索 。 现 在 我 们 就 去 看 看 吧 。 


13.5 深入 Grails 


可 惜 啊 ， 短 短 一 章 的 篇 幅 无 法 承载 Grails 框 架 的 所 有 内 容 ， 因 为 它 需 要 一 本 书 ! 在 这 一 节 ， 
我 们 再 为 新 加 入 Grails 阵 营 的 开发 人 员 讲 一 些 值得 探索 的 领域 : 

口 日 志 ; 

口 GORM: Grails 对 象 一 关系 映射 ; 

口 Grails 栖 件 。 

另外 ， 也 可 以 到 http:Wwww.grails.org 网 站 上 去 看 看 ， 上 面 有 关于 这 些 主题 的 基本 教程 。Glen 
Smith 和 Peter Ledbrook 写 的 Grails in Action( Manning，2009 ) 也 值得 仔细 阅读 。 

我 们 从 Grails 的 日 志 人 手 吧 。 


13.5.1 日 志 
Grails 的 日 志 功能 是 由 log4j 提 供 的 ， 在 grails-app/confConfig.groovy 文 件 中 配置 。 


DD 对 于 大 包 数 源码 来 说 都 是 如 此 ， 只 要 没 改 出 错 来 就 行 。 
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比如 说 ， 你 可 能 想 要 chapter13 包 中 的 代码 显示 WARN 消 息 ， 而 域 对 象 类 PlayerCcharacter 
只 显示 ERROR 消 息 。 要 满足 这 一 要 求 ， 可 以 把 下 面 这 段 代 码 放 到 log4j 的 配置 文件 Config.groovy 
文件 中 。 


J = | 
Wrn 'com. javaTdeveloper .chapterl3' 
error 'com.jJavaTdeveloper.chapterl3. ES 
iorg.codehaus .groovy .grails.web. servlet ' /i 控制 器 
} 


日 志 配 置 就 跟 你 过 去 用 log4j 的 log4j.xml 配 置 一 样 灵活 。 
接 下 来 我 们 会 看 看 Grails 中 的 对 象 关系 映射 技术 GORM。 


13.5.2 GORM: 对 象 关 系 映 射 


GORM 是 用 Spring/Hibernate 实 现 的 ， 这 是 Java 开 发 人 员 非 常熟 悉 的 技术 组 合 。 它 所 涵盖 的 功 
能 非常 广泛 ， 但 其 核心 功能 非常 像 Java 的 JPA。 

要 想 马 上 实验 一 下 它 的 持久 化 行为 ， 可 以 执行 如 下 命令 打开 Grails 控 制 台 : 

grails console 

还 记得 第 8 章 讲 的 Groovy 控 制 台 吗 ? 这 个 Grails 应 用 环境 跟 那个 非常 类 似 。 

首先 ， 我 们 保存 一 下 Playvercharacter 域 对 象 , 


import com.Javadeveloper.chapterl3.PlayerCharacter 
new PlayerCcharacter (lstrength:18, dexterity:15, charisma:15) .save() 


PlayerCharacter 保 存 好 后 有 很 多 种 办 法 可 以 读 取 它 。 最 简单 的 办 法 是 通过 Grails 添 加 到 域 
对 和 象 类 中 的 隐 含 id 属 性 取 回 可 写 的 完整 实例 。 在 控制 台 用 下 面 这 段 代 人 码 换 掉 前 面 那 段 并 执行 。 

import com.jJava7developer.chapterl3.PlayerCharacter 

def pc = PlayerCharacter .get (1) 

aSSeErt 18 == pe.strength 

要 更 新 对 和 象 ， 修 改 一 些 属性 然后 再 次 调用 save () 方 法 。 请 再 次 清空 控制 台 并 运行 下 面 这 段 
代码 。 

import com.java7Tdeveloper.chapteril3.PlayercCcharacter 

def pc = PlayerCcharacter .get (1) 

pe.strength = 5 

pe .Savel) 

pe = PlayerCharacter .get (1) 

assert 5 == pC.strength 


要 删除 对 象 请 用 delete() 方法 。 再 次 清空 控制 台 并 运行 下 面 的 代码 ， 删 除 


PlayerCharacter, 


import com.jJava7Tdeveloper.chapterl3.PFlayerCharacter 
def pc = PlayerCharacter .get!{1) 
pe.delete|) 
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GORM 具 备 完整 丰富 的 多 对 一 、 多 对 多 关系 声明 能 力 ， 以 及 其 他 我 们 熟悉 的 Hibernate/JPA 所 
支持 的 关系 声明 能 力 。 
现在 我 们 去 看 看 从 Rails“ 拿 来 ”的 插件 概念 。 


13.5.3 Grails 插件 


Grails 有 大 量 插件 ， 可 以 帮 开 发 人 员 完 成 常见 的 Web 开 发 任务 。 其 中 最 流行 的 插件 有 : 

口 Cloud Foundry Integration ( 用 于 将 应 用 部 署 到 云 服务 上 ); 

口 Quartz ( 用 于 计划 调度 ); 

口 Mail ( 用 于 处 理 电 子 邮 件 ) 

Twitter、Facebook ( 用 于 社交 网 络 集成 )。 

要 查看 有 哪些 插件 可 用 ， 请 执行 如 下 命令 : 

grails list-plugins 

然后 可 以 执行 grails plugin-info [名 称 ] 查 看 捅 件 的 更 多 信息 ， 用 感 兴趣 的 捅 件 名 称 替 换 
[名 称 ] 就 可 以 了 。 此 外 ， 也 可 以 访问 http://grails.org/plugins/ 深 入 了 解 这 些 插件 及 其 生态 系统 的 
信息 。 

要 安装 插件 ， 请 运行 grails install-plugin [名 称 ]， 用 要 安装 的 捅 件 名 称 赫 换 [名 称 ]。 
比如 说 ， 为 了 更 好 地 支持 日 期 和 时 间 ， 可 以 安 计 Joda-Time 搬 件 。 

grails inatall-plugin joda-time 

装 上 Joda-Time 插 件 后 ， 可 以 给 Playercharacter 加 上 LocalDate 属 性 。 把 下 面 的 ijmport 
语句 加 到 域 对 象 类 中 。 


import org.Joda.time.* 
import org.joda.time.contrib.hibernate.* 


把 下 面 这 个 属性 加 到 Playercharacter 中 ，。 

LocalDate timestamp = new LocalDate |) 

为 什么 这 跟 引 用 JAR 文 件 中 的 API 不 同 呢 ? 因为 Joda-Time 插 件 会 确保 该 类 型 跟 Grails 惯 例 优 
先 的 原则 若 容 。 这 就 是 说 Joda-Time 的 类 型 是 映射 到 数据 库 类 型 上 的 ， 并 且 完 全 支持 它 的 映射 和 
脚手架 人 处理 。 如 果 现 在 回 到 http://localhost:8080/pcgen grails/playerCharacter/list 页 面 中 ， 会 看 到 列 
出 了 日 期 。 

借助 插件 的 这 类 支持 ，Grails 开 发 人 员 可 以 用 很 短 的 时 间 构 建 出 数量 惊人 的 功能 。 

我 们 对 Grails 的 初次 拜访 结束 了 ,但 本 章 中 快速 Web 开 发 的 故事 还 没 讲 完 。 下 一 节 会 讨论 
Clojure 的 快速 Web 开 发 类 库 Compojure。 熟 悉 Clojure 的 开发 人 员 可 以 借助 它 用 简洁 的 Clojure 代 码 
还 速 构建 出 小 到 中 型 的 Web 应 用 。 
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13.6 Compojure 入 门 


开发 Web 最 致命 的 想法 就 是 把 什么 网 站 都 当成 Google 来 设计 。 对 于 Web 应 用 来 说 ， 过 度 设计 
和 设计 不 足 都 是 错误 的 。 

务实 而 优秀 的 开发 人 员 会 考虑 Web 应 用 的 上 下 文 ， 不 会 增加 任何 不 必要 的 复杂 性 。 认 真 分 析 
所 有 应 用 的 非 功能 需求 是 避免 构建 错误 的 关键 前 提 。 

Compojure 就 是 那 种 不 要 想 征服 世界 的 Web 框 架 。 对 于 Web 仪 表 板 、 操 作 监 控 ， 以 及 很 多 更 加 
注重 简单 性 和 开发 速度 、 而 不 是 大 规模 扩展 能 力 及 其 他 非 功 能 需求 的 简单 任务 来 说 ，Compojure 
是 非常 理想 的 选择 。 从 这 种 描述 中 你 应 该 能 猿 出 来 ，Compojure 介 于 多 语言 编程 金字 塔 的 领域 特 
定 层 和 动态 层 之 间 。 

在 这 一 节 我 们 会 搭建 一 个 简单 的 Hello World 应 用 ， 然 后 讨论 Compojure 把 Web 应 用 串 起 来 的 
简单 规则 。 在 用 这 些 规 则 措 建 示例 程序 之 前 ， 先 介绍 一 个 实用 的 Clojure HTML 类 库 ( Hiccup )。 

如 疼 13-6 所 示 ，Compojure 构 建 在 Ring 框 架 之 上 , Ring 框架 是 Clojure 连 接 到 Jetty Web 容 器 的 中 
间 件 。 但 使 用 Compojure/Ring 并 不 需要 对 Jetty 有 多 深入 的 了 解 。 我 们 先 用 一 个 简单 的 Hello World 
作为 Compojure 的 入 门 应 用 吧 。 


图 13-6 ”Compojure 和 Ring 


13.6.1 Hello Compolure 

开始 一 个 新 的 Compojure 项 目 非常 容易 , 因为 Compojure 跟 Leiningen 的 工作 流程 自然 融合 。 
果 你 还 没 装 Leiningen, 也 没 看 第 12 章 中 的 那 一 节 ,， 那 你 现在 就 应 该 去 把 这 两 件 事 做 了 ， 因 为 接 下 
来 的 内 容 要 求 你 熟悉 Leiningen。 

要 开始 一 个 新 项 目 ， 只 要 执行 一 个 普通 的 Leiningen 命 令 : 

lein new hello-compolure 

在 project.cljj 中 可 以 轻松 指明 项 目的 依 头 项 ,代码 清单 13-8 显 示 了 如 何在 project.clj 文 件 中 指定 
Hello World 项 目的 依赖 项 。 
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(defproject i eg 
:description "FIXME: write description" 
:dependencies [[org.clojure/clojure "1.2.1"] 
[compojure "0.6.2"])] 
:dev-dependencies [[lein-ring "0.4.0"]] 
:ring (:handler hello-compoijure .core/app}) 
宏 (defproject) 跟 第 12 章 那个 很 像 ， 不 过 多 了 两 个 元 数据 。 
口 :dev-dependencies 确 保 开 发 人 员 可 以 在 开发 时 使 用 1ein 命 令 。 稍 后 我 们 讨论 lein 
ring server 时 你 就 能 见 到 实例 了 。 
口 :ring 引 人 本 Ring 类 库 所 需 的 挂 钧 。 它 将 Ring 特 定 的 元 数据 映射 为 参数 。 
这 个 例子 中 给 Ring 传 人 本 一 个 :hanaler 属 性 , 看 起 来 它 希 望 得 到 hello-compojure.core 
命名 室 间 中 的 app 符 号 。 我 们 来 看 看 代码 清单 13-9 中 core.clj 中 对 它 的 声明 ， 以 便 找 出 它们 是 如 何 
相互 配合 的 。 


代码 清单 13-9 Compojure Hello World 中 简单 的 core.clj 文 件 
(ns hello-compojure .core 
(i:use compojure .core) 
(:require [compojure.route :as routel] 
[compojure.handler :as handler])})) 


(load "hello") 


(defroutes main-routegs 主 路 由 定义 
[GET TY/ [] lpage-hello-compojure)) | 


(route/resourceas "/") 
Ute/not-found "Paae nn Found" | 
(route,not -found age not found")) ] 注册 路 由 


(def app (handler/asite main-routes)) 本 

这 种 把 关联 信息 和 其 他 信息 保存 在 core.clj 中 的 惯例 非常 实用 。 当 有 URL 请 求 时 再 加 载 一 个 包 
含 对 应 函数 ( 页 面 函 数 ) 的 单独 文件 很 简单 。 这 确实 只 是 一 个 为 了 提高 可 读 性 ， 简 单 实现 关注 点 
分 离 的 惯例 。 

Compojure 使 用 了 一 组 规则 ， 称 为 路 由 ， 来 确定 如 何 处 理 接 人 的 HTTP 请 求 。 这 些 规 则 是 由 
Compojure 依 赖 的 Ring 框 架 提 供 的 ， 它 们 既 简单 又 实用 。 你 可 能 已 经 猜 出 来 了 ， 规 则 GEgm"y "告诉 
Web 服 务 带 如 何 处理 对 根 URL 的 SET 请求。 我 们 下 一 节 会 对 路 由 做 更 多 的 讨论 。 

为 了 完成 这 个 例子 的 代码 ， 还 需要 在 src/hello compojure 目 录 中 创建 hello.clj 文 件 。 在 这 个 文 
件 中 要 和 定义 一 个 如 下 所 示 的 页 面 函 数 (page-hello-compojurel) : 


Ins hello-=-compoure .COEGBj 


Idefn page-hello-compojure [] "<hlsHello Compojure</hl>") 
这 个 页 面 函 数 是 个 常规 的 Clojure 函 数 ， 它 会 返回 一 个 字符 串 作 为 HTML 文档 的 <bodayv> 标 签 
中 的 内 容 ， 而 这 个 文档 会 作为 响应 的 一 部 分 返回 给 用 户 。 
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单 的 操作 。 先 确保 所 有 依赖 项 都 装 


让 我 们 把 这 个 例子 跑 起 来 。 在 Compojure 中 这 是 个 十 分 简 
上 了 : 

ariel:hello-compoijure boxcats$ lein deps 

Downloading: org/clojure/clojure/l1.2.1/clojure-1.2.1 .pom from central 

Downloading: org/clojure/clojure/1.2.1/clojure-1.2.1.,.jar from central 

Copying 9 files to /Users/boxcat/projects/hello-compojure/lib 

Copying 17 files to /Users/boxcat/projecta/hello-compojure/lib/dev 


到 目前 为 止 一 切 虱 好 。 现 在 需要 把 它 跑 起 来 ， 可 以 用 Ring 提 供 的 ring server 方 法 。 


ariel:hello-compojure boxcats lein ring server 

2011-04=11 lg:02:48.596:INFO: :Logging to STDERR via org.mortbay .log.StdErrLog 
2011-04-11 18:02:48.615:INFO:; :JjJetty-6.1.26 

2011-04-11 18:02:48.743:INFO: :Started SocketConnector@0.0.0.0:3000 

started server on port 3000 


这 会 启动 一 个 简单 的 Ring/Jetty Web 服 务 器 ( 默认 端口 3000 ), 以 实现 快速 反馈 。 上 默认 情 况 下 ， 
这 个 服务 器 会 目 动 重 载 被 修改 的 文件 。 


米 告 ”需要 知道 开发 服务 器 的 重 拭 是 在 文件 这 一 层 实 现 的 。 这 意味 着 正在 运行 的 服务 器 可 能 会 
因为 重新 加 载 页 面 导 致 其 状态 被 冲 掉 (或 更 三， 被 部 分 冲 掉 )。 如 果 你 怀疑 发 生 了 这 种 情 
况 ， 并 因此 出 现 了 问题 ， 应 该 关 掉 服务 器 重新 启动 。 启 动 Ring/Jetty 很 快 ， 应 该 不 会 对 开 
发 时 间 有 太 大 影响 。 


如 果 用 浏览 器 访问 开发 机 上 的 3000 端 口 (或 本 机 http://127.0.0.1:3000 )， 应 该 会 看 到 页 面 中 显 
示 出 上 “Hello Compojure 。 


13.6.2 ”Ring 和 路 由 
我 们 来 看 看 如 何 配置 Compojure 应 用 的 路 由 。 路 由 的 定义 应 该 能 让 你 想到 一 种 领域 特定 语言 : 


(GET "/" [] (page-hello-compojure)})) 

这 些 路 由 规则 应 当 被 看 做 匹配 接 人 请 求 的 规则 。 其 构成 方式 非常 简单 

[<zHTTP methods <URL> zparams> <ACtions) 

口 HTTP 方 法 ， 通 常 是 GET 或 Pb5ST， 但 Compojure 也 支持 PUT、DELETE 和 HEAD。 如 果 要 匹配 

这 条 规则 ， 这 个 HTTP 方 法 必须 跟 传人 的 请 求 相 匹配 。 

口 URL， 请 求 对 应 的 URL。 如 果 要 匹配 这 条 规则 ， 这 个 URL 必 须 跟 传人 的 请 求 相 匹 配 。 

口 参数 ， 一 个 表示 参数 应 该 如 何 处 理 的 表达 式 。 很 快 我 们 就 会 对 它 展 开 讨 论 。 

D 动作 ， 与 这 条 规则 匹配 时 返回 的 表达 式 ( 通常 表示 为 传人 参数 的 函数 调用 )。 

对 这 些 规则 的 匹配 接 从 上 到 下 的 顺序 逐一 比 对 ， 直 到 找到 匹配 项 。Compojure 会 执行 第 一 个 
匹配 项 的 动作 ， 表 达 式 的 值 会 作为 返回 文档 <body> 标 签 中 的 内 容 。 

Compojure 中 规则 的 定义 很 灵活 。 比 如 说 , 创建 一 个 从 URL 中 提取 函数 参数 的 规则 非常 简单 。 
我 们 来 改 一 下 代码 清单 13-5 中 的 Hello World 路 由 : 


1 地 
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(defroutes main-routes 

(GET "/" [] (page-hello-compoijure)) 

(GET ["/hello/:fname", :fname #"[a-2ZA-Z]+" | 
sp [fname] (page-hello-with-name fname)) 


(route/resources "/") 
(route/not-found "Page not found")) 


这 个 新 规则 只 匹配 包含 /hello/< 名 称 > 的 URL。 其 中 的 名 称 只 能 包含 字母 ( 大写 、 小 写 或 大 
小 写 组 合 都 行 )， 这 是 由 Clojure 的 正则 表达 式 #" [a-zA-2z]+" 限 定 的 。 
如 果 匹 配 了 这 一 规则 ，Compojure 会 以 匹配 的 名 称 为 参数 调用 (page-hello-with-name)。 
函数 定义 非常 简单 : 
ldefn page-hello-with-name [fname] 
(atr "<hl>Hello from Compojure " fname "</h1i>")) 


只 有 非常 简单 的 应 用 才能 用 这 种 内 联 HTML, 否则 很 快 就 会 变 成 一 种 痛 。 好 在 有 Hiccup 模 块 
它 为 需要 输出 HTML 的 Web 应 用 提供 了 很 多 实用 的 功能 。 马 上 我 们 就 去 看 看 。 


13.6.3 Hiccup 


要 在 hello-compojure 应 用 中 挂 上 Hiccup， 需 要 做 三 件 事 : 

口 在 projectclj 上 加 上 依赖 项 ， 如 [hiccup "0.3.4"]; 

口 再 次 运行 1ein deps; 

口 重启 Web 容 器 。 

很 好 。 现 在 我 们 来 看 看 在 Clojure 内 部 怎么 用 Hiceup 写 出 更 好 的 HTML 形 式 。 

Hiccup 提 供 的 关键 形式 之 一 是 (hntml) 。 用 它 可 以 非常 直接 地 编写 HTML。 下 面 是 用 Hiccup 
重 写 的 (page-hello-with-name): 


(defn page-hello-html-name [fname] 
(html [i:hli "Hello from Compoiure " fname] 
[:div [:p "Paragraph text"]])) 


现在 这 些 骨 套 格式 的 HTML 标 签 看 起 来 很 像 Clojure 代 码 ， 所 以 把 它 放 到 代码 里 自然 多 了 。 
(html) 形式 以 一 个 或 更 多 的 《标签 ) 癌 量 为 参数 ， 并 且 标 签 的 檬 套 深度 不 受 限制 。 
接 下 来 ， 我 们 会 问 你 外 绍 一 个 稍微 大 一 点 儿 的 应 用 ， 一 个 给 水 猎 投 票 的 网 站 。 


13.7 ”我 是 不 是 一 只 水 猎 


互联 网 上 似乎 有 两 件 事 永 远 都 不 会 让 人 厌烦 ; 在 线 投票 和 可 爱 的 动物 图 片 。 有 个 创业 公司 想 
把 这 两 件 事 结 合 起 来 , 让 人 们 给 水 狂 图 片 投票 , 然后 靠 广 告 回 报 赚 钱 , 他 们 雇 了 你 。 勇敢 面 对 吧 ， 
这 时 部 还 算 不 上 是 创业 公司 所 尝试 过 的 最 傻 的 主意 。 
我 们 先 想 想 这 个 水 狂 投 票 网 站 所 需 的 基本 页 面 和 功能 : 
口 网 站 下 页 应 该 展示 两 张 水 杀 供 用 户 选择 ， 
口 用 户 应 该 能 给 目 己 至 欢 的 那 只 水 铬 投票 ; 
口 应 该 有 个 单独 的 页 面 多 许 用 户 上 传 水 猎 的 新 照片; 
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口 应 该 有 个 仪表 板 页 面 显 示 每 张 水 猫 图 片 的 当前 得 票 。 
图 13-7 中 展示 了 如 何 安排 构成 应 用 的 页 面 和 HTTP 请 求 。 


投票 es i 


和 


一 


图 13-7 “我 是 不 是 一 只 水 猫 ? ”的 页 面 流 
我 们 暂 不 考虑 该 应 用 的 非 功能 性 需求 。 
O 该 网 站 不 做 访问 控制 。 

口 对 新 上 传 的 水 多 图 片 文件 不 做 安全 检查 。 它 们 会 以 图 片 的 形式 在 页 面 上 显示 ,但 上 传 对 
象 的 内 容 或 安全 性 都 没有 经 过 检查 。 我 们 相信 用 户 ， 他 们 不 会 上 传 任何 不 合适 的 东西 。 
D 该 网 站 没有 持久 化 。 如 果 Web 容 器 崩溃 了 ， 所 有 投票 数据 就 都 没 了 。 但 在 应 用 启动 时 , 它 

会 扫描 硬盘 ， 预 先 填充 水 铬 图 片 的 存储 。 


尽管 我 们 会 在 这 一 章 中 介绍 其 中 的 重要 文件 ， 但 github.com 上 就 有 这 个 项 目 ， 你 可 能 会 发 现 
那个 更 好 用 。 


13.7.1 项 目 设 置 


要 开始 这 个 Compojure 项 目 ， 需 要 定义 基本 项 目 : 它 的 依赖 项 、 路 由 ， 还 有 一 些 页 面 国 数 。 
我 们 先 来 看 看 project.clj 文 件 ， 如 代码 清单 13-10 所 示 。 


代码 清单 13-10 ”项 目 project.clj 


(defproject am-i-an-otter "1.0.0-SNAPSHOT'" 
:descripticon "Am I an Otter or Not?" 
:dependencies [[org.clojure/clojure "1.2.0"] 
[org.clojure/clojure-contrib *1.2.0"] 
[compojure "0.6.2"] 
[hiccecup "0.3.4"] 


[log4j "1.2.15" :exclusions [javax.mail,/mail 


javax .jmas/jma 
com .sun .jamk/ jnmxtocls 


com.sun. jmx/jmxri]] 
[org .81f4j3/sl1f4j-api "1].5.6") 


[org .slf4j/s1f£4j-1094j12 "1.5.6"]] 
:dev-dependencies [llein-ring "0.4.0"]) 


:ring (|:handler am-i-an-otter.core/app}) 


1 二 
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这 个 文件 中 没什么 新 鲜 玩意 ， 除 了 log4j 类 库 ， 其 他 在 前 面 的 例子 里 都 有 。 
接 下 来 我 们 看 看 core.clj 文 件 里 的 连接 和 路 由 逻辑 ， 如 代码 清单 13-11 所 示 。 


i 
mp 


(ns am-i-an-otter. core 
(:uBe Compojure.core) 
(:require [compojure,route :as route] 
[compoijure.handler :as handler] 
[ring.middleware .multipart-params :as mp}])) 


(load "imports") | 
(load "otters-db") | 导入 函数 


(load "otters"™) a 
z 主 路 由 
(defroutegs main-routes 


(SET "/" [] (page-compare-otters)) 

(GET ["/upvote/:id", :id #"[0-3]+" ] [id] (page-upvote-otter id)) 
(GET "/upload" [] (page-start-upload-otter)) 

(GET "/votes" [] (page-otter-votes}) 


(mm/wrap-multipart -params i——— 
(POST “/add orter" Te 可 {atr (upload-otter req) 文件 上 传 处 理 程序 


(page=-start-upload-otter)))})) 


(route,resources ™/") 
(route/not-found "Page not found")) 


(def app 
(handler/site main-roOutes)) 


文件 上 传 处 理 程序 展示 了 一 种 新 的 参数 处 理 方式 。 我 们 在 下 一 小 节 还 会 展开 来 讲 ， 但 现在 ， 
可 以 把 它 看 做 “将 整个 HTTP 请 求 传 给 页 面 函数 处 理 ”。 

core.clj 中 的 关联 关系 让 你 可 以 看 清 哪 个 页 面 函 数 跟 哪个 URL 相 关 。 所 有 页 面 晴 数 都 以 page 打 
头 一 一 这 只 是 函数 命名 的 惯例 。 

nonsense es tr 


be 


Xu am- Fr -An- ey COre 
(:use compojure.core) 


(:use hiccup.core)) 
(defn page-compare-otters [] 水 帧 比较 页 面 


(let [otterl (random-otter), otter2 (random-otter)] 
(.info (get-=-logger) (str "Otterl = " Otterl " ; Otter2 = " 
ww Otter2 " 7 " Otter-pics)) 
(html [:hl "Otters say 'Hello Compojure!'"] 
[:p [:a {:href (gtr "/upvote/" ctterl) |】 
[:img {:sre (SEE "/img/" 
mb (get cotter-pics otter1i))}} ]]] 
[:p [:a {:hret (Str "“/upvote/" otter2)] 
[:img {:are lstr "/img/" 
= (get otter-pics otter2))} ]]] 
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[:P “Click ™ [:a {:href "/votes"} “Heren] 
" to see the votes for each otter")] 
[:p weliek " [:a {:href upload "here"] 


1 to upload a brand new otter"]))) ] 
| 处 理 投标 


(defn page-upvote-otter [id] 
(let [my-id id] 
(upvote-otter id) 
[str (html [:hl "Upvoted otter id=" my-id]) (page-compare-otters)))) 


(defn page-start-upload-otter |] 3 选择 水 名 上 传 
(html [:hl "Upload a new Dtter" |] 
[:p [:form {:action "/add otter" :method "POST" 设置 表单 


as :enctype "multipart/form-data") 
[:input {:name "file" :type "file" :size "20"°}] 
[:input {:name "gubmit" :type "submit" :value "submit"}]]] 
[:B "Or click " [:a {:href "/"} "here" ] " to vote on some otters"])) 
(defn page-otter-votes [] 一 一 
(let [] 显示 投票 结果 
(.debug (get-logger) (satr "Otters: " @otter-votes-r)) 
{html [:hl "otter Votegs" ] 
[:div#votes.otter-votes 
{for [x (keys ®otter-votegs-r)] 
[:p [:img {:arec lstr "/img/" lget otter-pics x}})}} ] 
mk (get ®@otter-votes-r XxX}]}}]})) 


代码 中 还 有 两 个 Hiccup 特 性 。 第 一 个 可 以 对 一 组 元 紊 进行 循环 , 在 这 儿 是 刚 上 传 的 水 猫 图 片 。 
Hiccup 在 下 面 的 代码 上 请 段 中 表现 得 非常 像 简单 的 模板 语言 (市 有 骨 人 的 (for) 形 态 ) 


[:divi#votes.otter-votes 
(for [x (keys Botter-votes-r)] 
[:p [:img (i:s8rc (str "/img/" (get otter-pics x))} ] 
ms (get Botter-votes-r XX)])] 


第 二 个 特性 是 :div#votes .otter-votes 语 法 。 这 是 指明 某 一 标签 的 ja 和 class 属 性 的 快 
捷 办 法 。 它 会 恋 成 HTML 标 签 <div class="otter-votes"” id="votes">。 开 发 人 员 可 以 俏 
此 把 最 可 能 由 CSS 使 用 的 属性 分 离 出 来 ， 不 会 让 HTML 结构 变 得 太 乱 。 

CSS 和 其 他 代码 ( 比如 JavaScript 源 文件 ) 通 常会 放 在 静态 内 容 目录 中 等 待 读 取 。 在 Compojure 
项 目 NS 目录 下 。 


rr , pp a Rd i A 0 Wd 用 村 a 可 癌症 Tim " > LA 广 站 记 ; a | 下 ”时 : 
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si 


水 了 投 村 这 不合 于 在 亲 构 上 有 要 消 。 pyr pe 这 是 错 
误 的 。 
应 用 程序 绝 不 应 该 用 GET 请 求 修改 服务 器 端的 状态 ( 比如 水 猎 的 投票 数 ), 因为 Web 浏 览 器 
在 觉得 服务 器 没有 响应 时 是 可 以 重 发 GET 请 求 的 (比如 当 请 求 进来 时 它 正 因为 垃圾 收集 而 暂停 
呢 )。 这 一 重 发 请 求 的 行为 可 能 会 导致 同一 水 猎 收 到 重复 投票 ， 可 实际 上 用 户 只 点 了 一 次 。 对 
于 电子 商务 应 用 来 说 ， 这 会 引发 灾难 | 
记 住 这 条 原则 : 有 意义 的 服务 器 端 状 态 绝 不 能 用 GET 请 求 修 改 。 
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我 们 已 经 看 过 了 关联 起 来 的 应 用 和 它 的 路 由 ,以 及 页 面 函 数 。 我 们 再 来 看 一 些 处 理 水 狂 书 
的 后 台 函 数 ， 继续 讨论 这 个 应 用 。 
13.7.2 ”核心 函数 


在 讨论 应 用 的 核心 功能 时 , 我 们 提 到 应 用 应 该 扫描 图 片 目 录 找 出 磁盘 里 已 有 的 水 猎 图 片 。 代 
码 清单 13-13 是 扫描 目录 并 进行 预 填充 的 代码 。 


(def otteaer-img-dir "resources/public/img/") 
ldef otter-img-dir-fq 
(astr (.getAbsolutePpath lIFile, "~.")}) /AT otter-img-dir)) 
[defn make-matcher [patternl] 
(,.getPathMatcher (FileSystems/getDefault) (str "glob:" pattern))}) 
(defn file-find [file matcherl] 村 PP 
(let [fname (.getName file (= (.getNameCount file) 1))] 如 果 匹 配 ， 返回 去 拍 
(if (and (net {nil? fname})) (.matches matcher fnamel ) 两 边 空 格 的 文件 名 
(.tostring fname) 
nil))) | 用 (tostring) 启 用 ;img 标 签 


(defn next-map-id [map-with-id] 下 一 个 
(+ 1 (nth {max (let [map-ids (keys map-with-=id)] $e 必 
(i£f (nil? map-ids} [0] map-ids))}) 0 }))) 水 赖 的 ID 


(defn alter-file-map [file-map fname] z ee 
(asgsoc file-map (next -map-id file-map) fname)) Fei 


(defn make-scanner [pattern file-map-r] 
(let [matcher [make-matcher pattern)] 返回 担 可 
(proxy [SimpleFileVvisitor] [] 


(visitFile [file attribsl] 要 ea : 
(let [my-file file, 在 所 有 文件 上 执行 的 
my-attrg attribs, 回调 函数 


file-name (file-find my-file matcher)] 
(.debug (get=logger) (str "Return from file-find " file-name})} 
(if {not {nil? file-name)) 
(dosync talter file-map-r alter-file-map file-name) file-map-r) 
nil) 
(.debug lget-logger) 
ms lstr "After return from file-find " @file-map-r)) 
FilevVvisitResult /CONTINUE)) 


(visitFileFailed [file EXC] (let [my-file file my-ex exc] 
(. info (get=logger) 
(str "Failed to access file " my-file " ; Exception: " my-ex)) 
FileVvisitResult /CONTINUE))))) 


(defn scan-for-otters [file-map-r] 
(let [my-map-r file-map-r] 
(Files/walkFileTree (Paths/get otter-img-dir-fq 
ss (into-array String []}) (make-scanner "* .Jpg" my-map-r)) 
my-map-r)) , 


| 设置 水 猫 图 片 


(def otter-pics (deref (scan-for-otters (ref |}})})}) 了 
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这 段 代码 的 人 口 是 (scan-for-otters)。 它 用 Java 7 中 的 Files 类 从 otter-img-dir-fa 
开始 遍历 文件 系统 ,并 返回 一 个 映射 , 这 里 用 了 一 个 简单 的 惯例 , 以-r 结 束 的 标记 名 称 表示 这 昨 
对 某 个 结构 的 引用 。 

省 历 文件 的 代码 是 simpleFileVisitor 类 (在 java.nio.file 包 中 ) 的 Clojure 代 理 , 这 个 


类 在 第 2 章 就 出 现 过 。 我 们 自行 实现 了 其 中 两 个 方法 ， (visitFile) 和 (visitFileFailed)， 
对 这 个 例子 来 说 足够 了 。 
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(def otter-votes-r (ref {})) 


(defn otter-exists [id] (contains? (set (kevys otter-picas)) id)) 


(defn alter-otter-upvote [vote-map id] 
(assgoc vote-map id (+ 1 (let [cur-votes (get wvote-map id})] 
(if (nil? cur-votes) 0 cur-votes))))) 


(defn upvote-otter [id] 
(if (otter-exists id) 
(let [my-id id] 
(.info (get-logger}) (str "Upvoted Otter " my-id)) 
(doayne (lalter otter-votes-r alter-otter-upvote my-id) 
mb Otter-votes-r)) 
(.info (get-logger) (str "Otter 上 id 人 Oot Found " otter-pice))})}) 


(defn random-otter [] lirand-nth (keys otter-pics))) 


(defn upload-otter [req] 赋予 随机 文件 名 
(let [new-id (next-map-id otter-pica), 
new-name (str (java.util .UUID/randomUUID) 


“ "Jpg"), 二 
tmp-file (:tempfile 提取 中 时 广 件 


us (get (|:multipart-params req) "file™"))})] 
({(.debug (get-logger) (SEE (.toString req) " ; New name = " 
mm New-name " ;} New id = " new-id)) 
(das/copy tmp-file (ds/file-etr | 复制 到 文件 系统 中 
ms (Btr otter-ijmg-dir new-name))})) 
(def otter-pics [assoc otter-pics new-id new-name)) 
(html [:hl "Otter Uploaded!"])}) 


在 (upload-ottezr) 图 数 中 处 理 的 是 完整 的 HTTP 请 求 映 射 。 其 中 有 很 多 信息 可 供 Web 开 发 
人 员 使 用 ， 不 过 有 些 可 能 是 你 已 经 熟悉 的 了 : 


{:remote-addr "127.0.0.1", 

:Scheme :http, 

:query-params {}, 

:Beagion {}, 

:form-params {【}, 

:multipart-params {"submit" "submit", “file" {:filename "otter kidas.jpg", 
:Bize 122017, :content-type "image/jpeg", :tempfile #<File /var/tmp/ 
upload 646a7Tdf3 12f5£51ff33 8000 00000000.tmp>}}, 

:Pequest-method :poBt, 

:query-string nil, 
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:route-params |}, 
:content-type "multipart/form-data; boundary=---- 
WebKitFormBoundaryviKZzehApamHrVFtO", 
:cookies {}, 
:uri "/add otter", 
: BeErver-nNname "ll27.0.0.1", 
:params {|:file {:Eilename "otter kids.jpg", :size 122017, :content-type 
"image/jpeg", :tempfile #<aFile /var/tmp/ 
upload 646a7dft3 12f5f51ff33 8000 00000000 .tmp>}, :Submit "submit"}), 
:headers {"user-agent" "Mozilla/5.0 (Macintosh; U; Intel Mac OS X 10 6 6; 
en-US) AppleWebKit/S34.16 (KHTML, like Gecko) Chrome/10.0.648.205 
Safari/534.16", "origin”" "http://127.0.0.1:3000", "accept-charset" "ISO- 
8859-1,utf-8;9=0.7,*j9q=0.3", "Accept" "application/xml,application/ 
xhtml+xml ,text/html ;gq=0.9,text/plain;g=0.8, image/png,*/*q=0.5", "host" 
"127.0.0.1:3000", "referer" "http://127.0.0.1:3000/upload", "content- 
type" "mltipart/form-data; boundary=---- 
WebKitFormaoundaryvEKZehApamWrVFtO", "cache-control" "max-age=0", 
"accept-encoding" "gzip.,.def late,sdch", "content-length" "122304™, 
"aAaccept -language" "en-US,en;q=0.8", "connection" "keep-alive"}, 
:Content-length 1223304， 
:SerVver-port 3000, 
:Character-encoding nil, 
:body #<Input org.mortbay.jetty.HttpParsers$Input@206bc833>| 


从 这 个 请 求 映 射 中 能 看 到 容器 已 经 把 上 传 的 文件 内 容 放 到 了 /vvarytmp 的 临时 文件 中 。 可 以 
通过 (:tempfile (get (:multipart-params reg) "file")) 访 问 相 应 的 File 对 象 。 然 后 
阐 单 地 用 clojure.contrib.duck-streams 中 的 (copy) 国 数 把 它 保存 到 文件 系统 中 。 

水 猎 投 票 不 大 , 但 它 是 一 个 完整 的 应 用 程序 。 在 本 节 开 头 提 出 的 功能 性 和 非 功 能 性 需求 的 限 
定 下 ， 它 的 表现 符合 我 们 的 预期 。 我 们 对 Compojure 及 一 些 相关 类 库 的 探索 就 到 此 为 止 了 。 


13.8 小结 


快速 Web 开 发 应 该 是 所 有 优秀 Java 开 发 人 员 都 能 做 的 事情 。 但 如 果 选 了 糟 烂 的 语言 或 框 
架 ， 很 快 你 就 会 落 在 Rails 和 PHP 这 种 非 Java/JVM 技 术 人 员 的 后 面 。 尤 其 是 静态 类 型 的 编译 型 
Java 语 言 ， 它 有 时候 不 是 做 Web 开 发 的 理 扯 选择 。 相 反 ， 选 对 了 语言 或 框架 。 就 可 以 在 保证 质 
量 的 前 提 下 快速 实现 新 功能 ， 助 你 攀 上 Web 开 发 食物 链 的 顶端 ， 可 以 针对 用 户 所 需 快 速 做 出 
反应 。 

优秀 的 Java 开 发 人 员 不 希望 扔 掉 强 大 灵 话 的 JVM。 幸 好 ， 随 着 JVM 上 的 其 他 语言 及 其 Web 框 
架 的 出 现 ， 你 可 以 留 着 它 了 ! 像 Grails 和 Compojure 这 样 的 动态 层 框架 提供 了 你 所 需要 的 快速 Web 
开发 能 力 。 

特别 是 Grails， 可 以 非常 迅速 地 措 建 一 个 完整 的 ( UI 到 数据 库 ) 原型 ， 然 后 开发 人 员 就 可 
以 用 强大 的 展示 层 技术 ( GSP )、 存 储 层 技术 ( GORM ) 和 一 大 堆 实用 的 插件 把 各 个 部 分 撑 
起 来 。 
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Compojure 可 以 很 自然 地 跟 Clojure 编 写 的 项 目 相 结 合 。 也 非常 适合 用 来 向 Java 或 其 他 语言 的 
项 目 中 添加 小 型 Web 组 件 , 比如 仪表 板 和 操作 控制 台 。 简洁 的 代码 和 快速 的 开发 能 力 是 Compojure 
的 主要 优势 。 

我 们 就 这 样 学 习 了 JVM 多 语言 编程 的 各 种 示例 , 走 到 了 各 章 的 结尾 。 在 最 后 一 章 , 我 们 会 把 
所 有 的 线索 都 抓 到 一 起 ,看 一 些 超前 的 知识 。 那 里 有 超出 我 们 现 有 经 验 之 外 的 挑战 ,但 现在 我 们 
掌握 的 工具 已 经 可 以 处 理 它们 了 。 
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保持 优秀 


本 章 内 容 

口 Java 8 对 开发 者 的 意义 
口 多 语言 编程 的 未 来 
口 并 发 性 的 发 展 分 方 问 
口 JVM 层 的 新 特性 


要 走 在 时 代 的 前 列 ， 优 秀 的 Java 开 发 人 员 总 是 应 该 对 即将 到 来 的 东西 保持 清醒 的 认识 。 本 书 
最 后 一 章 会 讨论 几 个 在 我 们 看 来 指引 Java 语 言及 平台 未 来 发 展 方向 的 主题 。 

因为 我 们 既 没 有 TARDIS 也 没有 水 唱 球 , 所 以 本 章 的 内 容 主要 集中 在 据 我 们 所 知已 经 在 开发 
的 语言 特性 和 平台 修改 上 。 也 就 是 说 这 只 能 是 当下 的 观点 , 客气 的 说 法 是 这 在 茶 种 程度 上 来 说 算 
是 科 约 作品 。 

撰写 本 书 过 程 中 我 们 所 讨论 的 观点 只 是 代表 将 来 的 一 种 可 能 。 事 情 如 何 发 展 还 有 待 时 间 验 
证 。 毫 无 疑问 的 是 , 在 某 些 重要 方式 上 事情 的 发 展会 跟 我 们 此 处 的 讨论 有 所 不 同 , 并 且 以 非常 有 
趣 的 方式 到 达 那 一 点 。 通 常 都 是 这 样 。 

让 我 们 先 去 看 看 第 一 个 主题 吧 ， 快 速 浏览 一 下 很 可 能 出 现在 Java 8 中 的 一 些 主要 特性 。 


14.1 对 Java 8 的 期 行 


2010 年 秋 ，Java SE 执行 委员 会 南 议 决定 执行 B 计 划 。 这 个 决定 是 尽快 发 布 Java 7， 并 把 一 些 
主要 特性 延迟 到 Java 8 中 。 这 一 结论 是 在 对 社区 进行 广泛 征询 和 投票 后 得 出 的 。 

在 Java 7 中 发 起 的 某 些 特性 已 经 被 推 到 Java 8 中 了 , 还 有 些 特性 已 经 缩减 了 范围 以 便 为 将 来 的 
特性 打下 基础 。 在 这 一 节 中 ， 我 们 会 对 一 些 期 望 Java 8 突出 的 特性 做 简要 介绍 ， 和 包括 那些 被 延迟 
的 特性 。 在 这 一 阶段 ， 没 有 什么 是 板 上 钉 钉 的 ， 特 别 是 语法 。 所 有 示例 代码 都 只 是 初步 构想 ， 可 
能 跟 Java 8 的 最 终 写法 差别 很 大 。 欢 迎 来 到 风口 浪 尖 1 


DD TARDIS 是 英国 科幻 电视 剧 《神秘 博士 》( Docior yo ) 中 的 时 间 机 器 和 宇宙 飞船 ， 是 时 间 和 空间 相对 维度 ( Time 
and Relative Dimension In Space ) 的 缩写 。 一 一 译 者 注 
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14.1.1 _ lambda 表达 式 ( 闭 包 ) 


将 在 Java 8 中 构建 的 Java 7 特性 中 最 有 代表 性 的 是 MethodHandles 和 invokedynamic。 它们 
本 身 是 非常 实用 的 特性 ( 在 Java 7 中 ，invokedynamic 主 要 用 在 语言 和 框架 实现 上 )。 

在 Java 8 中 , 这些 特性 是 将 lambda 表 达 式 引入 Java 语 言 的 基础 。 可 以 这 样 理解 ,lambda 表 达 式 跟 
前 面 在 备 选 语言 中 讲 的 函数 字面 值 类 似 ， 它们 也 能 用 来 解决 我 们 在 前 面 重点 强调 的 那 类 问题 。 

就 Java 8 的 语法 而 言 ， 还 需要 决定 lambda 表 达 式 在 代码 中 如 何 表 示 。 但 其 基本 特性 已 经 确 
下 来 了 ， 所 以 我 们 先 来 看 看 基本 的 Java 8 语法 ， 如 代码 清单 14-1 所 示 。 


代码 清单 14-1 在 Java 中 用 lambda 表 达 式 实现 Schwartzian 变 换 
public List<T> SCchwarzi(Lisc<T> X，Mapper<T，V> £f) | 
return XxX.maplw -> new PaircT,Vs(lw, £f .maplw)}) 
.SoOrted((l,r) -> l1.hashed.compareTo{r .hashed)) 
-map(l -> l.orig) .into(lnew ArrayList<T>()); 


定 


| 

schwartz() 方 法 看 起 来 应 该 眼熟 ， 它 是 在 10.3 节 中 用 闭 包 实现 的 Schwartzian 变 换 。 代 码 清 
单 14-1 展 示 了 Java 8 中 lambda 表 达 式 的 下 列 基本 语法 : 

口 在 lambda 表 达 式 前 部 有 个 参数 列表 ; 

口 组 成 lambda 表达 式 的 主体 代码 块 用 括号 括 起 来 ; 

口 用 稍 头 (-> ) 来 分 隔 参 数列 表 和 lambda 表 达 式 的 主体 ; 

口 参数 列表 中 参数 的 类 型 是 可 推断 的 。 

第 9 章 中 Scala 的 困 数 字面 值 和 这 个 写法 很 像 ， 所 以 这 种 请 法 应 该 不 会 让 你 觉得 特别 陌生 。 代 
码 清单 14-1 中 的 lambda 表 达 式 非常 短 ， 全 都 只 有 一 行 。 实 际 上 ，lambda 表 达 式 是 可 以 包含 多 行 代 
码 的 ， 其 主体 甚至 可 以 很 大 。 经 过 初步 分 析 ,， 那些 适 于 改造 成 lamhbda 表 达 式 的 代码 改造 后 的 长 度 
都 应 该 在 1 到 $ 行 之 间 。 

代码 清单 14-1 中 还 介绍 了 另外 一 个 新 特性 。 变 量 x 的 类 型 是 List<T>。 我 们 在 x 上 调用 了 方法 
map()。map () 方 法 接受 了 一 个 lambda 表 达 式 作为 其 参数 。 停 ! List 接 口 根本 就 没有 map () 方 法， 
并 且 在 Java 7 及 之 前 都 不 存在 lambda 表 达 式 。 

我 们 来 仔细 看 看 这 个 问题 是 如 何 解 决 的 。 

1. 扩展 和 默认 方法 

我 们 所 面临 的 问题 本 质 是 : 怎么 向 已 有 接口 中 添加 方法 以 使 其 “lambda 化 ”而 又 不 破坏 其 向 
后 菲 容 性 ? 

答案 来 自 于 Java 的 一 个 新 特性 : 扩展 方法 。 它 可 以 为 没有 提供 扩展 方法 的 接口 实现 提供 一 个 
可 用 的 默认 方法 。 

这 些 默认 的 方法 实现 必须 在 接口 本 身 内 部 定义 。 比 如 跟 List 搭 配 的 AbstractList， 跟 Map 
措 配 的 AbstractMap， 跟 Dueue 措 配 的 AMbstractoueue。 这 些 类 是 为 各 自 的 接口 存放 新 的 扩展 
方法 默认 实现 的 理想 之 所 。Java 内 置 的 集合 类 是 扩展 方法 和 lambda 化 的 主要 应 用 场景 ,但 看 起 来 
这 种 模型 在 最 终 用 户 代 码 中 也 适用 。 
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扩展 方法 无 需 打破 向 后 兼容 性 就 可 以 让 发 布 后 的 接口 得 到 进化 。 开 发 人 员 可 以 借 此 带 着 
lambda 表 达 式 为 老 旧 的 API 注 人 新 的 活力 。 但 lambda 表 达 式 对 于 JVM 来 说 是 什么 呢 ? 是 对 象 吗 ? 
如 果 是 ， 它 们 的 类 型 是 什么 ? 

2. SAM 转 换 

mbda 表 达 式 提供 了 一 种 紧 壮 的 办 法 ,可 以 声明 少量 内 联 代 码 并 将 其 作为 数据 传递 。 也 就 是 
说 lambda 是 一 个 对 象 ， 就 像 我 们 在 本 书 第 三 部 分 中 对 lambda 表 达 式 和 函数 字面 值 的 解释 一 样 。 具 
体 说 来 ， 你 可 以 把 lambda 表 达 式 当做 object 的 一 个 子 类 ， 它 没有 参数 ( 因此 也 没有 状态 )， 只 有 
一 人 有 

还 有 一 种 理解 这 个 问题 的 方式 : 通过 术语 SAM ( Single Abstract Method， 单 例 抽 象 方法 )。 
SAM 的 概念 在 各 种 Java API 中 都 有 体现 ， 是 一 种 常见 主题 。 很 多 API 中 都 有 只 声明 了 一 个 单 例 方 
法 的 接口 。Runnable、Comparable、Callable 和 ActionListener 之 类 的 监听 问 都 只 声明 名 
了 一 个 方法 ， 因 此 都 算 SAM 类 。 

在 刚 开 始 用 lambda 表 达 式 时 , 可 以 把 它们 当做 语法 糖 一 一 为 给 定 接口 编写 匿名 实现 的 简便 写 
法 。 过 一 段 时 间 后 , 你 可 以 掌握 更 多 的 困 数 式 技 术 , 甚至 可 能 会 从 Scala 或 Clojure 中 把 自己 喜欢 的 
技巧 引信 Java 代 码 中 。 学 习 函 数 式 编程 是 个 循序 渐进 的 过 程 : 从 集合 的 映射 、 排 序 和 过 滤 技 术 开 
始 学 起 ， 然 后 慢 慢 向 外 开 疆 拓 土 。 

现在 让 我 们 进入 下 一 个 大 主题 模块 化 编程 ， 它 正在 Jigsaw 项 目的 支持 下 如 火 如 茶 地 展开 。 


14.1.2 ”模块 化 (拼图 Jigsaw) 


外 理 classpat 了 h 有 时 毫 无 疑问 是 不 太 理 想 的 .有 
所 周知 的 问题 ， 

DJRE 目 身 的 规模 就 比较 大 ; 

D JAR 文 件 提倡 整体 式 部 署 模型 ; 

有 些 繁琐 上 且 极 少 会 用 到 的 类 仍然 必须 加 载 ; 

局 动 慢 ; 

D classpath 是 脆弱 的 野 曾 ， 并 且 跟 机 器 上 的 文件 系统 结合 得 过 于 紧密 ; 

口 classpath 基 本 上 是 一 个 局 平 化 的 命名 空间 ; 

口 JAR 不 具有 固有 的 版 本 ; 

口 即便 逻辑 上 没有 关联 的 类 之 间 也 有 复杂 的 相互 依赖 关系 。 


目 绕 着 JAR 文 件 和 classpath 构 建 的 生态 系统 有 些 众 
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要 解决 这 些 问 题 ,需要 一 个 新 的 模块 系统 .但 要 先 解决 架构 上 的 问题 。 其 中 最 重要 的 如 图 14-1 
所 示 。 
用 户 层 模块 化 模块 化 平台 


国王 芭 
(并 让 aol) 


图 14-1 模块 化 系统 的 架构 选择 


我 们 是 应 该 引导 VM 然后 青 使 用 用 户 层 模 块 化 系统 【( 比如 OSGi )， 还 是 应 该 彻底 迁移 到 模块 
化 平 全 上? 

后 一 种 方式 需要 启动 一 个 能 支持 模块 的 最 基本 的 “内 核 ” VM ， 然 后 根据 启动 应 用 程序 的 
需要 添加 特定 的 模块 。 这 要 求 对 VM， 以 及 JRE 中 很 多 现 有 的 类 做 颠覆 性 的 修改 ,但 潜在 收益 
更 大 。 

口 JVM 应 用 程序 的 启动 时 间 可 以 跟 shell 和 脚本 语言 相 媲美 。 

口 能 显著 降低 应 用 程序 部 着 的 复杂 性 。 

口 针对 性 的 Java 安 装 所 占用 的 次 源 可 以 显 者 碱 少 ( 对 硬盘 、 内 存 和 安全 性 都 有 积极 影响 )。 

如 果 你 不 需要 CORBA 或 RMI， 就 不 用 装 ! 
口 可 以 以 更 加 灵活 的 方式 升级 Java 安 闭 。 如 果 在 Collections 中 发 现 了 一 个 严重 的 bug, 只 有 那 
个 模块 需要 升级 。 

撰写 本 书 时 看 起 来 Jigsaw 项 目 会 选择 第 二 种 方式 。 但 在 它 发 布 并 能 投入 使 用 之 前 ， 还 有 很 长 
的 路 要 走 。 下 面 是 一 些 仍 在 讨论 的 重要 问题 : 

口 平台 或 应 用 的 正确 发 布 单 元 是 什么 ? 

口 是 不 是 需要 一 种 跟 包 和 JAR 都 不 同 的 新 结构 ? 

这 一 设计 决策 的 影响 极为 重要 : Java 无 处 不 在 ， 因 而 这 种 模块 化 的 设计 要 渗透 到 所 有 地 方 。 
它 也 要 支持 跨 OS 平 台 。 

Java 平 台 最 起 码 要 能 在 Linux、Solaris、Windows、Mac OS XX、BSD Unix 和 AIX 上 部 署 模块 化 
应 用 。 这 些 平 台中 有 些 有 需要 Java 模 块 集成 的 包 管 理 器 ( 比如 Debian 的 apt、Red Hat 的 mpm， 以 及 
Solaris 包 )。 而 其 他 平台 ， 比 如 Windows， 没 有 可 供 Java 使 用 的 包 管 理 系统 。 

这 一 设计 还 有 其 他 的 限制 . 不 过 这 一 领域 已 经 有 一 些 成 熟 的 项 目 了 : 比如 Maven 和 Ivy 这 样 的 


1 入 
和 


Ei 14.2 多 语言 编程 365 
依 琅 项 管理 系统 ,还 有 发 起 倡议 的 OSGi。 新 的 模块 化 系统 应 该 尽 一 切 可 能 跟 现 有 项 目 集成 ， 即 
便 在 完全 的 集成 和 羔 容 被 证 明 不 可 能 之 后 ， 也 应 该 提供 一 个 顺畅 的 升级 途径 。 
不 管 将 来 会 怎样 ，Java 8 的 发 布 应 该 会 给 Java 应 用 程序 的 交付 和 部 署 带 来 革命 性 的 变化 。 
我 们 去 看 看 JDK 8 应 该 给 JVM 的 其 他 公民 带 来 的 一 些 特 性 ， 包 括 我 们 在 前 面 研 究 的 那些 
语言 。 


14.2 多 语言 编程 


从 第 5 章 开 始 , 你 已 经 无 数 次 见证 了 JVM 作 为 语言 运行 时 平台 的 奇妙 ,。 第 ] 章 介绍 的 OpenJDK 
项 目 在 Java 7 的 发 布 周 期 中 成 了 Java 的 参考 实现 。 非 常 有 趣 的 是 JVM 已 经 发 展 成 了 一 个 语言 无 关 
的 、 真 正 支持 多 语言 编程 的 虚拟 机 。 

特别 是 随 着 Java 7 的 发 布 ，Java 语 言 老 失 了 在 VM 上 的 特权 。 平台 上 的 所 有 语言 现在 都 一 视 同 
仁 。 因 此 人 们 对 添加 之 于 备 选 语言 非常 重要 、 而 对 Java 本 身 只 有 边际 效益 的 VM 特性 表现 出 了 强 
烈 的 兴趣 。 

这 一 工作 是 在 达 苍 奇 机 ( Da Vinci Machine ) 子 项 目 中 开展 的 ， 这 一 项 目 也 叫做 mlvm ( 多 语 
言 VM )。 在 这 一 项 目 中 培育 出 的 特性 会 被 引入 源码 主干 中 。5.5 节 中 的 invokedynamic 就 是 这 样 
的 例子 ,但 还 有 很 多 对 非 Java 语 言 非常 实用 的 其 他 特性 。 也 有 需要 解决 的 问题 。 

我 们 先 来 看 看 这 些 语言 特性 中 的 第 一 个 : 不 同 的 语言 运行 在 同一 个 JVM 中 彼此 进行 无 障碍 交 


流 的 办 法 。 
元 对 象 协议 


14.2.1 语言 的 互 操作 性 及 

语言 平等 是 向 了 不 起 的 多 语言 编程 环境 迈 出 的 重要 一 步 , 但 一 些 棘 手 的 问题 仍然 存在 。 其 中 
主要 是 不 同 的 语言 有 不 同 的 类 型 系统 。Ruby 的 字符 串 是 可 修改 的 ， 而 Java 的 不 可 修改 。Scala 把 所 
有 东西 都 当成 对 象 ， 即 便 是 在 Java 里 作为 原始 类 型 的 实体 也 是 如 此 。 

处 理 这 些 差异 , 并 为 同一 JVM 内 的 不 同 语言 提供 更 好 的 交互 和 互 操作 方式 , 是 目前 尚未 解决 
的 问题 ， 也 是 正在 积极 处 理 有 望 近 期 解决 的 问题 。 

想象 一 个 将 来 要 做 的 Web 应 用 : 其 核心 部 分 可 能 是 Java 代 码 , Web 部 分 是 用 Compojure 写 的 ( 即 
Clojure )， 所 用 的 JSON 处 理 类 库 是 用 纯粹 的 JavaScript 写 的 ,而 你 想 用 ScalaTest 中 一 些 很 酪 的 TDD 
功能 来 对 它 进行 测试 。 

这 形成 了 一 个 JavaScript、Clojure 、Scala 和 Java 彼 此 之 间 都 会 直接 调用 的 局 面 。 对 JVMi 语 言 能 
够 互 操作 并 以 一 种 标准 的 方式 调用 彼此 对 象 的 需求 会 随 着 时 间 逐 渐 增 强 , 社区 内 的 广泛 共识 是 需 
要 一 种 元 对 象 协议 ( Metaobject Protocol, MOP ), 以 便 所 有 这 些 语 言 都 能 以 一 种 标准 的 方式 工作 。 
MOP 可 以 看 做 是 一 种 在 代码 内 捕 述 特定 的 语言 如 何 实现 面向 对 象 及 相关 问题 的 办 法 。 

要 实现 这 一 目标 ， 我 们 需要 想 想 那些 能 让 某 种 语言 中 的 对 象 能 在 另外 一 种 语言 中 使 用 的 办 
法 。 一 种 简单 的 方式 是 把 它 转换 成 其 他 语言 中 的 本 地 类 型 (或 甚至 在 外 部 运行 时 中 创建 一 个 新 的 “| 
“影子 ”对 象 )。 这 种 办 法 简单 ， 但 有 严重 的 问题 
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口 所 有 语言 必须 都 有 一 个 通用 的 “ 主 ” 接口 (或 超 类 ), 语言 内 的 所 有 类 型 都 必须 实现 它 ( 比 
如 JRuby 中 的 IRubyobject ); 

口 如 果 用 影子 对 象 ， 那 会 增加 很 多 内 存 分 配 ， 性 能 也 会 受 影响 。 

相反 ,我 们 可 以 考虑 为 外 部 运行 时 构建 一 个 服务 作为 入 口 。 这 一 服务 会 提供 一 个 接口 ,用 一 
运行 时 可 以 通过 该 接口 对 外 部 运行 时 中 的 对 象 执行 标准 操作 ， 比 如 : 

D 在 其 他 语言 运行 时 中 创建 一 个 新 对 得 并 返回 对 它 的 引用 | 

口 访问 ( 获取 方法 或 设置 方法 ) 外 部 对 象 的 属性 ; 

口 调用 外 部 对 象 上 的 方法 ， 并 返回 纺 采 ; 

口 将 外 部 对 象 转换 成 不 同 的 相关 类 型 ; 

口 访问 外 部 对 象 的 其 他 能 力 ， 对 一 些 语言 来 说 可 能 和 方法 调用 的 语义 有 所 不 同 。 

在 这 样 的 系统 中 , 可 以 通过 在 外 部 运行 时 上 调用 navigatoz 来 访问 外 部 方法 或 属性 。 调 用 者 
需要 提供 一 种 办 法 来 标识 要 访问 的 方法 : someMethod。 通 常 是 个 字符 串 ， 但 某 些 情况 下 也 可 能 
是 MethodHandle。 

navigator.callMethod(lsomeQbject, someMethod, paraml, param2, ...); 

要 让 这 种 办 法 起 作用 ， 所 有 协作 语言 运行 时 中 的 navigator 接 口 必须 都 一 样 。 实 际 上 , 语言 之 
间 的 真实 联系 很 可 能 是 用 invokedynamic 建 立 起 来 的 。 

接 下 来 我 们 去 看 看 多 语言 JVM 和 Java 8 的 模块 化 子 系统 组 合 起 来 是 个 什么 样子 。 


14.2.2 多 语言 模块 化 


随 着 Jigsaw 和 平台 模块 化 的 出 现 , 不 仅仅 是 Java 才 会 从 模块 化 中 受益 ( 并 需要 参与 进来 ) 其 
他 语言 也 能 加 人 其 中 有 所 表现 。 

可 以 想象 ，navigator 接 口 及 其 辅助 类 很 可 能 会 成 为 一 个 模块 ， 对 某 一 非 Java 语 言 运行 时 的 支 
持 将 会 由 一 个 或 多 个 模块 实现 。 图 14-2 中 展示 了 这 一 模块 系统 看 起 来 是 什么 样子 。 


如 你 所 见 ， 我 们 可 以 用 模块 系统 搭建 包含 多 种 语言 的 应 用 程序 。Clojure 模 块 提供 基本 的 
Clojure 平 各 ，Compojure 模 块 引信 运行 webapp 所 需 的 组 件 ， 包 括 特定 版 本 的 JAR， 在 别处 运行 时 
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这 些 JAR 可 能 会 用 不 同 的 版 本 。Scala 及 其 XML 也 出 现 了 , 为 了 实现 Scala 和 Clojure 之 间 的 互 操作 ， 
Navigator 模 块 也 出 现 了 。 
在 下 一 节 中 , 我 们 会 讨论 非 Java 语 言 在 平台 上 的 爆炸 性 涌现 所 推动 的 另 一 个 编程 趋势 : 并 发 。 


14.3 未 来 的 并 发 趋 势 


20 世 纪 的 语言 不 一 定 能 充分 发 挥 21 世 纪 的 硬件 的 作用 。 我 们 之 前 讨论 的 内 容 已 经 多 次 暗示 了 
这 一 点 。 在 第 6 童 讨 论 唱 体 管 数量 增长 的 摩尔 定律 时 (6.3.1 节 ), 由 于 一 个 非常 重要 的 原因 ,我 们 
当时 仅 简 要 讨论 了 一 下 。 那 就 是 摩尔 定律 、 性 能 和 并 发 之 间 的 相互 作用 ， 这 也 是 我 们 的 第 一 个 
主题 。 


14.3.1 多 核 的 世界 


尽管 晶体 管 数量 的 爆炸 性 增长 跟 预 测 一 样 , 但 内 存 的 访问 速度 却 没 能 跟 上 。 在 20 世 纪 90 年 代 
和 21 世 纪 头 几 年 ， 芯 片 设计 者 用 大 量 晶体 管 解决 相对 较 慢 的 主 存 问题 。 

就 像 第 6 章 讨论 的 ， 这 可 以 确保 有 稳定 的 数据 流 供 核心 处 理 。 但 这 根本 是 一 场 输 掉 的 战斗 
在 晶体 管 上 提升 的 速度 变 得 越 来 越 边 际 化 。 这 是 因为 过 去 用 的 技术 ( 比如 指令 级 并 行 和 投机 式 执 
行 ) 现在 已 经 把 容易 提升 的 速度 榨 光 了 ， 投 机 性 变 得 越 来 越 强 。 

最 近 这 些 年 , 业界 已 经 把 注意 力 转 到 了 用 晶体 管 在 每 个 芯片 上 提供 更 多 处 理 器 内 核 。 现 在 几 
乎 所 有 笔记 本 或 台式 机 都 至 少 是 双核 的 ，4 核 和 8 核 也 很 常见 。 在 更 高 端的 服务 器 上 ， 可 以 找到 6 
核 或 8 核 的 芯片 ， 整 机 能 达到 32 ( 或 更 多 ) 个 核心 。 多 核 世界 就 在 这 里 ， 要 充分 发 挥 它 的 作用 ， 
需要 以 串 行 处 理 更 少 的 风格 编程 。 那 需要 得 到 语言 和 运行 时 环境 的 支持 。 


14.3.2 ”运行 时 管理 的 并 发 


我 们 已 经 看 到 了 未 来 并 发 编程 的 开始 。 在 Scala 和 Clojure 中 , 我们 讨论 了 与 Java 的 线程 和 锁 模 
型 有 很 大 差异 的 并 发 观点 : S$cala 的 actor 模 型 和 Clojure 的 软件 事务 型 内 存 方式 。 

Scala 的 actor 模 型 允许 在 运行 的 代码 块 之 间 发 送 消息 ,而 这 些 代码 可 能 运行 在 完全 不 同 的 核心 
上 (甚至 有 人 允许 actor 远 程 运 行 的 扩展 )。 这 就 是 说 代码 完全 是 按 以 actor 为 中 心 的 方式 编写 的 ， 因 
此 在 多 核 机 器 上 扩展 非常 简单 。 

跟 Scala actor 一 样 ，Clojure 中 的 代理 填补 了 相同 的 生态 位 ”, 但 Clojure 中 还 有 只 能 在 一 个 内 存 
事务 中 修改 的 共享 数据 ( refs ) 一 一 软件 事务 型 内 存 机 制 。 

在 这 两 种 并 发 中 ， 都 能 见 到 一 种 新 概念 的 萌芽 ， 由 运行 时 ( 而 不 是 开发 人 员 ) 管理 并 发 。 尽 
管 JVM 提 供 了 线程 调度 的 底层 服务 ， 但 它 没 有 提供 管理 并 发 程序 的 高 屋 结 构 。 

这 个 缺陷 在 Java 语 言 中 能 看 出 来 ， 导 致 Java 程 序 员 革 本 上 在 用 JVM 的 底层 模型 。 


由 生态 位 ( Ecological niche )， 又 称 小 生 填 、 生 态 区 位 、 生态 柄 位 或 是 生态 使 位 ， 是 一 个 物种 所 处 的 环境 以 及 其 本身 | 14 
生活 习性 的 总 称 。 每 个 物种 都 有 自己 独特 的 生态 位 ， 区 别 于 其 他 物种 ， 一 一 译 者 注 = 
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PFA 人 opp pr 
我 们 才 对 可 变数 据 、 上 默认 共享 状态 和 用 协作 和 锁 强 制 排他 执行 这 种 模型 的 问题 有 所 察觉 。 但 发 布 
Java 1.0 的 工程 师 没有 这 种 福利 。 从 很 多 方面 来 说 , Java 在 并 发 上 的 首次 尝试 都 是 我 们 取得 今天 
这 种 成 就 的 基础 。 


现在 有 大 把 的 代码 撒 在 外 面 ， 再 为 Java 做 一 种 全 新 的 机 制 来 强制 推行 ， 还 要 跟 现 有 代码 无 缝 
交互 ， 这 非常 困难 。 所 以 大 部 分 注意 力 都 放 在 了 为 非 Java 的 JVM 语 言 找寻 新 的 并 发 出 路 上 。 这 些 
语言 有 了 两 个 重要 特性 ; 

口 以 JMM 为 底层 模型 ; 

口 跟 Java 相 比 有 “全 新 设计 ”的 声言 运行 时 ， 可 以 提供 不 同 的 抽象 层 ( 并 且 强 制 性 更 强 )。 

在 VM 层面 出 现 更 多 的 并 发 支持 也 不 是 不 可 能 (下 一 节 会 讨论 到 ), 但 目前 来 看 主流 还 是 在 以 
JMM 为 基础 的 新 语言 上 做 创新 ， 而 不 是 修改 底层 的 基础 线程 模型 。 

在 JDK 8 及 以 后 的 版 本 中 ， 上 肯定 能 见 到 JVM 的 某 些 区 域 会 发 生变 化 。 其 中 的 一 些 变 化 顺延 了 
Java 7 的 ijnvokedynamic， 这 也 是 我 们 要 讨论 的 下 一 主题 。 


14.4 ” JVM 的 新 方 问 


我 们 在 第 1 章 介 绍 VMSpec( JVM 规 范 ), 这 一 文档 确切 指明 了 作为 JVM 标 准 实 现 的 VM 必须 
遵守 的 行为 准则 。 当 引信 新 行为 时 ( 比如 Java 7 的 ijnvokedynamic )， 所 有 实现 都 必须 升级 以 支 
持 新 功能 。 

在 这 一 节 里 , 我 们 会 谈 到 那些 已 经 在 讨论 并 有 原型 的 各 种 修改 最 终 实现 的 可 能 性 。 这 项 工作 
是 在 OpenJDK 项 目 中 开展 的 ， 该 项 目 是 Java 参 考 实现 的 基础 ， 也 是 Oracle JDK 的 起 点 。 除 了 对 规 
范 的 可 能 修改 ,我 们 也 会 涉及 对 OpenJDK/Oracle JDK 代 码 的 显著 改动 。 


14.4.1 VM 的 合并 


在 Oracle 收 购 了 Sun 人 和 公司 之 后 , 它 就 拥有 了 两 款 非 常 强 的 Java 虚 拟 机 : HotSpot VM ( Sun 带 的 ) 
和 JRockit ( 之 前 收购 的 BEA 禹 的 

Oracle 很 快 就 决定 不 再 同时 维护 两 个 VM 来 浪费 资源 ， 要 把 它们 合并 起 来 。HotSpot VM 被 选 
作 基 础 ，JRockit 特 性 会 在 将 来 发 布 Java 时 并 慎 地 引进 。 


pp 官 ; 方 名 称 ， 尽管 VM 办 和 Java 社 区 天 部 分 孝 ee yy 。 
它 也 确实 挺 吸 引 人 ， 但 还 是 要 看 Oracle 的 营销 部 门 同 不 同意 。 


只 自在 读书 @3 
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所 以 这 对 咱 开 发 人 员 来 说 ， 这 有 什么 关系 呢 ? 你 现在 用 的 VM ( 很 可 能 是 HotSpot VM ) 将 来 
会 增加 很 多 新 特性 ， 包 括 ( 但 不 限于 ) 下 面 这 些 ， 
口 去 掉 PermGen ， 能 防止 一 大 类 跟 类 加 载 有 天 的 前 泪 ; 
D 加 强 JMX 代 理 的 支持 ， 能 让 你 对 运行 的 VM 有 更 多 深入 的 了 解 ; 
口 新 的 JIT 编 译 方式 ， 从 JRockit 中 引入 新 的 优化 ; 
口 任务 兵制， 提供 有 助 于 对 生产 型 应 用 进行 调 优 和 分 析 的 先进 工具 。 这 些 工 具 中 有 些 可 能 
是 需要 付费 的 外 加 JVM 组 件 ， 不 包含 在 免费 下 载 的 发 布 包 中 。 


Fit 要 li) el il 攻 持 | > PR 有 i W 村 
就 像 6.5.2 节 说 的 ， 类 的 用 教 据 当前 rp 内 存 区 里 es ) 它 很 快 
就 会 被 填 满 ， 特别 是 对 于 那些 在 运行 时 会 创建 大 量 类 的 非 Java 语 言 和 框架 而 言 。PermGen 区 不 
回收 ， 耗 光 之 5 有 关 人 员 eae 二 本 六 直入 认 汪 种 涛 全 科 轿 兴 ， 


让 重 梦 一 般 的 “java.lang. space” 消 息 永远 地 成 为 过 去 。 


还 有 很 多 的 小 改进 全 都 是 为 了 让 VM 更 小 、 更 快 、 更 灵活 。 假 定 HotSpot 上 大 约 已 经 投入 了 1000 
人 年 的 工作 量 ， 跟 投入 工作 量 更 多 的 JRockit 结 合 起 来 形成 的 VM 前 景 一 定 更 加 光明 。 

除了 合并 VM， 还 有 大 量 的 新 特性 正在 制作 中 。 其 中 之 一 就 是 可 能 会 增加 称 为 协同 程序 的 并 
发 特性 。 


14.4.2 ”协同 程序 


Java 和 JVM 语 言 程 序 员 了 解 最 多 的 并 发 形式 就 是 多 线程 。 它 依靠 JVM 的 线程 调度 服务 在 处 理 
器 核心 上 启动 和 停止 线程 ,但 线程 没 办 法 控制 这 个 调度 。 出 于 这 一 原因 ， 多 线程 被 称 为 “抢占 式 
多 任务 "， 因 为 调度 器 可 以 抢占 正在 运行 的 线程 ， 迫 使 它 放 弃 对 CPU 的 控制 。 

协同 程序 的 基本 思想 是 允许 执行 单元 部 分 参与 控制 对 它们 的 调度 。 具体 来 说 ,协同 程序 会 像 
普通 线程 那样 运行 ,直到 它 遇 到 了 一 个 “退位 ”指令 。 这 会 导致 协同 程序 把 自己 挂 起 ， 并 人 允许 另 
一 个 协同 程序 继续 在 它 的 地 盘 运 行 。 当 原来 的 协同 程序 再 次 得 到 机 会 运行 时 , 它 会 从 退位 之 后 的 
下 一 条 语句 继续 向 下 执行 ， 而 不 会 从 方法 开始 的 地 方 。 

因为 这 种 多 线程 的 方式 徘 正在 运行 的 协同 程序 的 相互 协作 , 间或 将 运行 机 会 退让 给 其 他 协同 
程序 ， 这 种 多 线程 处 理 敏 称 为 “协作 式 多 任务 ”。 

关于 协同 程序 如 何 工作 的 确切 设计 仍然 处 于 热烈 讨论 的 阶段 ， 没 有 哪个 是 肯定 要 被 采纳 的 。 
一 个 可 能 的 模型 是 在 一 个 单 例 共享 线程 (或 类 似 于 java .util .concurrent 里 的 线程 池 ) 中 创 
建 和 调度 协同 程序 ， 如 图 14-3 所 示 。 

正在 执行 协同 程序 的 线程 可 能 会 被 系统 内 的 其 他 任何 线程 抢占 , 但 JVM 线 程 调度 器 不 能 强迫 
协同 程序 退位 。 也 就 是 说 ， 以 相信 执行 池 中 所 有 其 他 协同 程序 为 代价 ,协同 程序 就 可 以 控制 什么 
时 候 切 换 上 下 文 。 


1 于 
OEEEE 
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协同 程序 线程 地 普通 线程 


图 14-3 ”一 种 可 能 的 协同 程序 模型 


这 种 控制 意味 着 协同 程序 之 间 的 同步 可 以 做 得 更 好 。 多 线程 代码 必须 构建 复杂 的 锁 策 略 来 保 
护 数据 ， 但 它 很 脆弱 ， 因 为 上 下 文 切 换 可 能 随时 都 会 发 生 。 这 是 我 们 在 4.1 节 讨论 的 并 发 类 型 安 
全 问题 。 相 较 而 言 ， 协 同 程序 只 要 确保 退位 点 数据 的 一 致 性 ， 因 为 它 知 道 其 他 任何 时 候 上 自己 都 不 
会 被 抢占 。 

这 个 折 中 的 额外 担保 是 以 相信 其 他 线程 为 交换 条 件 的 ， 这 是 对 某 些 线程 编程 问题 的 有 益 补 
充 。 一 些 非 Java 语 言 已 经 开始 支持 协同 程序 ( 或 与 之 很 贴近 的 概念 “纤维 ”)， 特 别 是 Ruby 和 较 新 
版 的 JavaScript。 在 VM 层面 增加 协同 程序 的 文 持 (但 不 一 定 是 对 Java 语 言 ) 会 对 可 以 使 用 协同 程 
序 的 语言 有 很 大 帮助 。 

在 可 能 会 实现 的 VM 修 改 中 ， 最 后 要 讨论 的 是 “元 组 "”， 这 个 VM 特 性 提案 对 性 能 敏感 的 计算 
室 间 可 能 会 产生 很 大 的 影响 。 


14.4.3 元 组 


在 当今 的 JVM 里 ， 所 有 数据 项 不 是 原 妨 类 型 就 是 引用 类 型 ( 可 能 是 引用 对 象 或 数组 )。 比 较 
复杂 的 类 型 只 能 在 类 里 定义 ,并 传递 对 这 些 新 类 型 实例 对 象 的 引用 。 这 是 一 个 简单 而 又 相当 优雅 
的 模型 ， 过 去 一 直 为 Java 服 务 得 很 好 。 

但 要 构建 高 性 能 系统 ,这 个 模型 就 会 暴露 几 个 缺陷 ,尤其 是 在 游戏 和 人 金融 软件 这 样 的 应 用 中 ， 
过 到 这 个 简单 模型 局 限 性 的 情况 十 分 常见 。 可 以 解决 这 个 问题 的 办 法 之 一 就 是 来 用 元 组 。 

元 组 ( tuple ) 有 时 称 为 值 对 象 ， 是 能 在 原始 类 型 和 类 之 间架 起 桥梁 的 语言 结构 。 像 类 一 样 ， 
用 元 组 可 以 定义 包含 原始 类 型 、 引 用 类 型 和 其 他 元 组 的 日 定义 复 末 类 型 。 像 原始 类 型 一 样 ， 在 将 
它们 传递 给 方法 (或 从 方法 中 传递 出 来 )， 保 存在 数组 和 其 他 对 象 中 时 ， 用 的 是 整个 值 。 如 果 你 
熟悉 C (或 .NET ) 环境 ， 可 以 把 它们 看 做 结构 ( struct ) 的 等 价 物 。 

我 们 来 看 一 个 例子 : 一 个 现 有 的 JavaAPI。 


public class MyInputSstream 1 
public woid write(byte[], int off, int len).; 


| 


1 于 
OEEEE 
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这 让 用 户 可 以 将 特定 数量 的 数据 写 到 数组 中 的 特定 位 置 , 很 实用 , 但 它 设计 得 并 不 好 ,在 理 
想 的 面向 对 象 屁 界 , 偏 移 和 长 度 应 该 被 封装 在 数组 内 , 并 且 无 论 是 用 户 还 是 方法 的 实现 者 都 应 该 
不 用 再 单独 跟踪 额外 的 信息 。 

实际 上 , 在 引入 NIO 时 ByteBuffer 就 封装 了 这 些 信息 。 可惜 这 不 是 白 来 的 , 从 ByteBuffer 
中 创建 新 切片 需要 分 配 一 个 新 对 象 , 这 会 给 垃圾 收集 子 系统 造成 压力 。 尽 管 大 多 数 垃圾 收集 器 都 
非常 擅长 收集 短命 的 对 和 象 , 但 在 看 叶 率 非常 高 的 延迟 敏感 环境 中 ,这 种 分 配 操作 会 累加 并 最 终 导 
致 应 用 出 现今 人 无 法 接受 的 暂停 。 

如 朵 我 们 能 定义 一 个 保存 数组 引用 、 偏 移 和 长 度 的 值 对 象 ( 也 就 是 元 组 ) 类 型 silice 会 发 生 
什么 呢 ? 在 代码 清单 14-2 中 ， 我 们 会 用 新 的 tuple 关 键 字 来 表示 这 个 新 概念 。 


代码 清单 14-2 ”作为 元 组 的 数组 切片 
public tuple Slice 1 
private int offset:; 
private int length; 
private bytel[l] array: 


public byte get(int i) 1 
return array [offset + i1]; 
| 
} 


这 个 切片 的 构造 结合 了 原始 类 型 和 引用 类 型 的 很 多 优势 : 
口 slice 值 可 以 复制 到 方法 中 , 也 可 以 从 方法 中 复制 出 来 ， 就 跟 手 工 传递 数组 的 引用 和 int 
值 一 样 有 效 ; 
口 Slice 元 组 在 退出 方法 后 会 被 清理 掉 ( 因为 它们 跟 值 类 型 一 样 ); 
口 对 偏 移 和 长 度 的 人 处 理会 干净 地 封装 在 元 组 中 。 
在 日 第 编程 中 有 很 多 类 型 会 从 元 组 的 使 用 中 受益 ,比如 带 有 分 子 和 分 母 的 有 理 数 、 带 有 实 部 
部 的 复数 ， 或 者 由 ID 和 领域 标识 引用 的 用 户主 档 ( 献 给 那些 MMORPG 迷 们 )。 
在 处 理 数 组 时 元 组 也 能 对 性 能 有 所 提升 。 现 在 的 数组 中 要 放 同 质 的 数据 值 集 合 一 一 要 么 是 原 
始 类 型 ， 要 么 是 引用 类 型 。 在 使 用 数组 时 ， 元 组 允许 我 们 对 内 存 的 布局 做 更 多 的 控制 。 
来 看 一 个 例子 ， 一 个 以 原始 类 型 1ong 为 键 的 简单 散 列 表 。 


public class MyHashTable | 
Brivate Entryl[) entries; 


二 有 a 
IE 


public class Entry | 
private long key, 
private Object value; 


} 

在 当前 的 JVM 化 身 中 ，entries 数 组 中 只 能 放 Entry 实 例 的 引用 。 调 用 者 每 次 查找 表 中 的 
key， 在 用 传人 的 值 与 相关 Entry 实 例 的 key 比 较 之 前 ， 必 须 把 Entry 宇 例 解 引用 。 

当 用 元 组 实现 时 ， 则 可 以 在 数组 内 展开 Entry 类 型 ， 因 此 能 够 省 掉 访 问 key 产 生 的 解 引用 开 | 
铺 。 图 14-4 展 示 了 当前 情况 ， 以 及 使 用 元 组 后 得 到 的 改善 。 14 
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图 14-4 JVM 数 组 与 元 组 


在 考察 元 组 的 数组 时 ， 有 用 元 组 得 到 性 能 优势 的 关键 之 处 也 变 得 更 加 清晰 。 我 们 在 第 6 章 讨 
论 过 , 大 多 数 应 用 程序 代码 的 性 能 都 是 由 一 级 绥 存 的 命中 率 决 定 的 。 在 图 14-4 中 , 如 果 使 用 元 组 ， 
扫描 散 列 表 的 代码 效率 会 更 高 。 它 不 用 再 承担 额外 的 缓存 读 取 就 能 得 到 key 值 。 这 就 是 元 组 取得 
性 能 优势 的 本 质 一 一 程序 员 在 展开 内 存 的 数据 时 可 以 得 到 更 优 的 空间 局 部 性 。? 

对 可 能 出 现在 Java 和 JDK 8 中 的 新 特性 ,我们 的 讨论 就 到 此 为 止 了 。 其 中 有 和 多少 能 变 成 现实 
也 只 能 等 到 快要 发 布 时 才 知 道 。 如 果 你 对 特性 的 演进 感 兴趣 ， 可 以 加 入 OpenJDK 项 目 和 Java 
Community Process， 参 加 这 些 特性 的 开发 活动 。 如 果 你 对 它们 还 不 熟悉 ， 请 找到 这 些 项 目 并 看 看 
如 何 加 人 。 


14.5 ”小结 


Java 8 会 紧 随 Java 7 的 脚步 。 它 会 满载 着 各 种 改进 而 来 , 让 开发 人 员 可 以 更 加 高 效 地 为 现代 化 
硬件 编写 代码 ， 无 论 是 最 小 的 舱 人 式 设 备 还 是 最 大 的 主机 。 

一 种 语言 解决 所 有 编程 问题 的 神话 已 经 破灭 了 。 为 了 搭建 有 效 的 解决 方案 ， 比 如 高 并 发 的 交 
易 系 统 ， 开 发 人 员 需 要 学 习 能 跟 核 心 Java 代 码 互 操作 的 新 语言 。 

随 厦 多核 硬件 和 OS 持续 提供 高 度 并 行 的 编码 架构 ， 并 发 仍然 是 热门 话题 。 想 要 解决 海量 数 
据 、 复 杂 运 算 或 高 速 应 用 ， 这 是 一 个 必须 跟 进 的 领域 。 

Java VM 被 看 做 是 当前 最 好 的 虚拟 机 。 因 为 优秀 的 Java 开 发 人 员 很 可 能 会 开辟 高 性 能 计算 等 
新 领域 ， 所 以 有 必要 密切 关注 未 来 的 发 展 态势 。 

如 你 所 见 ， 这 里 正在 发 生 着 巨变 ! 我 们 认为 Java 生 态 系 统 正在 经 历 一 次 规模 宏大 的 涅 保 ， 再 
过 几 年 ， 优 秀 的 Java 开 发 人 员 必 将 光彩 烟 烟 。 


DD 空间 局 部 性 ( spatial locality )， 如 果 程 序 访问 某 个 存储 器 地 址 后 ， 又 在 较 短 时 间 内 访问 临近 的 存储 器 地 址 ， 则 程 
序 具有 良好 的 空间 局 部 性 。 两 次 访问 的 地 址 越 接 近 ， 空 间 局 部 性 越 好 。 一 一 译 者 注 
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java7developer: 源码 安 


在 读 一 本 新 的 技术 书 时 , 我 们 都 喜欢 真 刀 真 枪 地 用 代码 做 练习 。 它 能 帮 我 们 正确 理解 书 中 的 
内 容 ， 光 靠 读 代 码 达 不 到 那 种 效果 。 

本 书 源码 可 在 www.manning.com/evans/ 或 www.java7Tdeveloper.com/ 作 处 下 载 。 我们 会 把 你 放 代 
码 的 位 置 称 为 SBOOK _CODE.。 

本 书 中 所 有 源码 都 放 在 java7developer 项 目 中 。 它 混合 了 Java、Groovy 、Scala 和 Clojure 的 源码 ， 
以 及 它们 的 支持 类 库 和 资源 。 它 不 是 那 种 典型 的 Java 项 目 ， 你 需要 按照 这 个 附录 中 的 指令 来 构建 
它 ( 即 编译 源码 和 运行 测试 ), 我 们 会 用 Maven 3 来 执行 各 种 构建 周期 目标 , 比如 compile 和 test。 

先 来 看 看 java7developer 项 目的 源码 布局 。 


A.1 java7developer 的 源码 结构 
java7developer 项 目的 结构 遵守 我 们 在 第 12 章 介绍 的 Maven 规 范 ， 因 此 布局 方式 如 下 所 示 : 


java7Tdeveloper 
|== 1ib 

|=-- pom .Xml 

| -- sample posix build.properties 
|=-- sample windows build,properties 


=-= javairdeveloper 
“== Chapter8 


| 

| 

| 

\ 

| 

| Java7Tdeveloper 
= chapterl 
| 

| 

| 


-= TESOUICEeS 
== Scala 
-= Com 


山 也 可 在 图 灵 社 区 【 wwwituring com.cn ) 本 书 网 页 免费 注册 下 载 。 一 一 编者 注 
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-= java7Tdeveloper 
| “== chaptergs 
= 一 七 号 号 蕊 
“一 = Java 
“== COM 


= Java7developer 
“一 一 Chapterl 


-= Scala 
= COM 
“=-- dava7Tdeveloper 
“== Chapterg 


- target 

按 它 的 规范 ，Maven 把 主 代 码 和 测试 代码 分 开 了 。 它 还 为 其 他 需要 包含 在 构建 中 的 文件 区 了 
个 特殊 的 resources 目 录 ( 比如 日 志 记 录 的 log4j.xml、Hibermate 配 置 文件 以 及 其 他 类 似 资源 ),. Maven 
的 构建 脚本 是 pom.xml 文 件 ， 附 录 E 中 有 对 它 的 详细 讨论 。 

Scala 和 Groovy 源 码 跟 Java 源 码 的 目录 结构 一 样 ， 只 是 Java 的 根 目录 是 java， 而 它们 的 根 目录 
分 别 是 scala 和 groovy。Java、Scala 和 Groovy 在 Maven 项 目 中 可 以 排 排 坐 ， 和 有 睦 相 处 ，Clojure 的 源 
码 处 理 起 来 稍 有 不 同 。Clojure 大 多 数 都 是 通过 一 个 交互 式 环境 处 理 的 (所 用 的 构建 工具 也 不 辣 ， 
叫 Leiningen )， 所 以 我 们 只 是 提供 了 一 个 clojure 目 录 ， 用 来 存放 Clojure 源 码 ， 做 练习 的 时 候 可 以 
复制 到 Clojure REPL 中 。 

在 Maven 构 建 运行 之 前 不 会 创建 target 目 录 。 构建 产生 的 所 有 类 、 工 件 、 报告 和 其 他 文件 都 会 
出 现在 这 个 目录 下 。 

lib 目 录 中 放 了 些 类 库 文 件 ， 以 防 Maven 不 能 访问 互联 网 下 载 所 需 类 库 。 

看 看 项 目 结构 ,让 自己 熟悉 一 下 各 章 的 源码 都 放 在 哪里 。 一 且 搞 清楚 源码 的 位 置 ， 就 可 以 安 
装 和 配置 Maven 3 了 。 


A.2 下 载 并 安装 Maven 


可 以 到 http://maven.apache.org/download.html 下 载 Maven。 在 第 12 章 的 例子 中 ， 我 们 用 的 是 
Maven 3.0.3。 如 里 你 用 的 是 *nix 操 作 系 统 , 请 下 载 apache-maven-3.0.3-bin.tar.gz， 如 果 是 Windows， 
则 下 载 apache-maven-3.0.3-bin.zip。 文 件 下 载 完 成 后 ， 只 要 选 好 目录 把 文件 解压 untar/gunzip 或 
unzip ) 就 行 了 。 


警告 ”中 很 多 Java/JVM 相 关 软 件 的 安装 一 样 ， 在 安装 Maven 的 目录 名 称 中 也 不 要 有 空格 ， 否 则 
可 能 会 出 现 PATH 和 CLASSPRATH 错 误 。 比 如 说 ， 如 果 你 用 的 是 Windows 操 作 系 统 ， 不 要 把 
Maven 装 在 C:\Program Files\Mavem\ 这 样 的 目录 中 。 


在 下 载 和 解压 完成 后 ， 接 下 来 就 是 设置 M2_HOME 环 境 变量 。 在 *nix 系 统 中 ， 需 要 加 一 些 下 面 
这 梓 的 东西 


1 入 
和 


M2 HOME=/opt /apache-maven-3.0.,3 


在 Windows 系 统 中 是 这 样 的 : 


M2 HOME=C:\apache-maven-3.0.3 

你 可 能 在 想 :“ 为 什么 是 M2_HOME 而 不 是 M3_HOME? 毕竟 这 是 Maven 3， 对 不 对 ? ”这 是 因为 
Maven 的 开发 团队 真 的 很 想 跟 得 到 广泛 应 用 的 Maven 2 保持 兼容 。 

Maven 融 要 Java JDK 才 能 运行 。1.5 之 后 的 版 本 都 行 ( 当然 ， 到 这 一 阶段 ,你 已 经 狠 好 JDK 1.7 
了 )。 还 需要 确保 环境 变量 JAVA_HOME 已 经 设置 好 了 一 一 如 果 已 经 装 好 Java 了 ， 那 这 个 环境 变量 
可 能 已 经 设置 好 了 。 还 需要 能 在 命令 行 中 的 任何 地 方 执行 Maven 相 关 的 命令 ， 所 以 应 该 在 PATH 
中 加 上 M2_HOME/bin 目 录 。 在 *nix 系 统 中 ， 需 要 加 一 些 下 面 这 样 的 东西 : 

PATH=$PATH: $M2_ HOME/bin 

在 Windows 系 统 中 是 这 样 的 : 

PATH=%PATH% ; $M2 HOMES\bin 


现在 可 以 带 着 -version 参 数 执行 Maven ( mwn )， 以 确保 基本 安装 可 用 。 


应 该 能 见 到 Maven 输 出 了 类 似 下 面 这 种 信息 : 


Apache Maven 3.0.3 (rli075438; 2011=-02-28 17:31:09*0000) 

Maven home: C:\apache-maven-3.0.3 

Java Vergsion: 1.7.0, vendor: Oracle Corporation 

Java home: C:\Java\jdkl .7.0\jre 

Default locale: en GB, platform encoding: Cpl252 

OS name: "windows xp", version: "5.1", arch: "x86", family: "windows”" 


如 你 所 见 ，Maven 批 量 输出 了 很 多 实用 的 配置 信息 ， 这 样 你 就 知道 Maven 及 其 依赖 项 在 你 的 
平台 上 都 OK 了 。 


提示 ”主流 IDE ( Eclipse、JIntelliJ 和 NetBeans ) 都 支持 Maven， 所 以 热 悉 了 Maven 在 命令 行 中 的 
使 用 方法 之 后 ， 可 以 直接 切换 到 IDE 集 成 的 版 本 。 

现在 Maven 已 经 装 好 了 ， 该 去 看 看 用 户 设置 放 在 哪里 了 。 为 了 触发 用 户 设置 目录 的 创建 ， 需 
要 确保 Maven 搬 件 已 经 下 载 并 安装 好 了 。 执 行 起 来 最 简单 的 是 帮助 ( Help ) 插件 。 

mvn help:system 

这 会 下 载 、 安 装 、 并 运行 帮助 插件 ， 它 给 出 的 信息 要 比 mvn -version 还 多 。 还 会 确保 .m2 
目录 已 经 创建 好 了 。 知 道 用 户 设置 放 哪 里 很 重要 ,因为 有 那么 几 次 你 可 能 需要 编辑 用 户 设 置 ,， 比 
如 让 Maven 能 用 在 一 个 代理 服务 器 后 面 。home 目 录 ( 我 们 会 用 $HOME 表 示 ) 中 能 看 到 表 A-1 中 列 
出 的 目录 和 文件 。 
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表 A-1 Maven 用 户 目 录 和 文件 ” 
是 材 解 释 

$HOME/m2 包含 Maven 用 户 配置 的 隐藏 目录 

SHOME/m2/settings xml 包含 用 户 特定 配置 的 文件 。 在 这 个 文件 中 可 以 指定 旁 路 代理 、 私 有 资源 库 以 及 定制 
Maven 行 为 的 其 他 信息 

SHOME/.m2/repository/ Maven 的 本 地 资源 库 。 当 Maven 从 Maven Central ( 或 其 他 的 远程 Maven 资 源 库 ) 下 
载 插件 或 依赖 项 时 , 它 会 在 本 地 资源 库 中 保存 一 份 副本 。 在 你 用 instal1 目 标 安装 
本 地 依 款项 时 也 蚌 这 样 。 这 样 Maven 就 可 以 用 本 地 副本 ， 而 不 用 每 次 都 去 下 载 


注意 ， 用 .m2 目录 还 是 因为 要 保持 跟 Maven 2 的 向 后 兼容 ( 而 不 是 你 认为 的 .m3 目录 )。 
现在 已 经 波 好 了 Maven， 也 知道 用 户 配 置 在 哪里 了 了， 可 以 开始 构建 java7developer 本。 


A.3 构建 java7developer 


这 一 节 会 从 几 个 一 次 性 步骤 开始 ， 为 构建 做 好 准备 ”。 这 包括 手动 安装 类 库 、 重 命名 属性 文 
件 并 编辑 它 ， 指 向 Java 7 的 本 地 安装 。 

然后 就 是 最 常见 的 Maven 构 建 周 期 目标 (clean、compile 和 test )。 第 一 个 构建 期 目标 
( clean ) 用 来 清理 上 一 次 构建 遗留 下 来 的 工件 。 

Maven 的 构建 脚本 是 POM ( Project Object Model， 项 目 对 象 模 型 ) 文件 。 这 些 POM 文 件 就 是 
XML 文件 ， 每 个 Maven 项 目 或 模块 都 有 一 个 对 应 的 pom.xml 文 件 。POM 文 件 即将 会 对 备 选 语言 
供 支 持 ， 满 足 你 所 需要 的 更 强 的 灵活 性 ( 很 像 Gradle )。 

要 用 Maven 执 行 构建 , 可 以 让 它 执行 一 个 或 几 个 表示 特定 任务 ( 比如 编译 源码 、 运 行 测 试 等 ) 
的 目标 。 目 标 全 部 都 是 绑 定 到 默认 构建 周期 中 的 ， 所 以 如 果 要 求 Maven 运 行 一 些 测试 ( 如 mvn 
test )， 它 会 在 试图 运行 测试 之 前 把 主 源码 和 用 于 测试 的 源码 都 编译 一 下 。 简 言 之 ， 它 会 迫使 你 
遵循 正确 的 构建 周期 。 

让 我 们 从 一 个 一 次 性 的 准备 任务 开始 吧 。 


A.3.1 一 次 性 的 构建 准备 工作 


要 成 功 运行 构建 ， 需 要 先 重 命 名 属性 文件 并 编辑 。 如 果 在 读 12.2 节 时 你 没 这 么 做 ， 请 转 到 
$BOOK_CODE 目 录 下 ， 将 sample <os> build.properties 文 件 (os 是 你 的 操作 系统 ) 另存 为 
build.properties ， 修 改 jak .javac.fullpath 属 性 ， 将 其 值 指 向 Java 7 的 本 地 安装 。 这 可 以 保证 
Maven 构 建 Java 代 码 时 能 选择 正确 的 JDK。 

准备 工作 做 好 了 ， 可 以 运行 clean 目 标 了 ， 执 行 构建 时 应 该 总 是 把 它 包 括 在 内 。 


中 向 Sonatype 禾 敬 ， 引 自 Movenm the Complete Reference 在 线 手册 ( www.sonatype.com/Request/Book/Maven-The- 
Complete-Reference ), 
加 尽管 Maven 构 建 工 具 最 近 有 改进 ， 并 且 也 支持 多 语言 编程 ， 但 还 是 有 些 差距 。 


只 自 在 读书 @ 


WWwwW .Zizidiary .com 


附录 A java7developer: 源码 安装 377 


A.3.2 clean 


clean 目 标 仅仅 是 把 target 目 录 删 掉 。 要 看 实际 效果 ， 请 切换 到 $BOOK _ CODE 目 录 并 执行 
clean 目 标 。 


cd $BOOK CODE 
mvr clean 


这 时 候 ， 你 会 看 到 控制 台中 满 是 Maven 下 载 各 种 插件 和 第 三 方 类 库 的 输出 信息 。Maven 需 要 
这 些 插件 和 类 上 库 运 行 目标 ， 它 默认 从 Maven Central ( 这 些 工 件 的 主要 在 线 资 源 库 ) 下 载 。 
java7developer 项 目 还 配置 了 另外 一 个 资源 库 ， 以 便 可 以 下 载 asm-4.0.jar 文 件 。 


注意 Maven 偶 尔 也 会 为 其 他 目标 执行 这 个 任务 ， 所 以 在 执行 其 他 目标 时 看 到 它 “ 下 载 互联 网 ” 
不 要 大 惊 小 怪 。 这 些 东 西 它 只 会 下 载 一 次 。 


除了 “正在 下 载 ……” 的 信息 ， did 


[INFO] -=-=-=-=-=-=-=-====-==-=====================m========n====-======-======= 一 一 
[INFO] BUILD SUCCESS 


INFO] -=------------------------------------------------- =- a 
INFO] Total time: 1.7038 


[INFO] Finished at: Fri Jun 24 13:51:58 BST 2011 
[INFO] Final an ee 
LINPFO) =e-===-=ameeem 和 erseretes ee 


如 果 clean 目 标 失 败 了 ， 很 可 能 是 代理 服务 器 阻止 你 访问 Maven Central 使 你 无 法 下 载 插件 
和 第 三 方 类 库 。 要 解决 这 个 问题 ， 只 需 修改 4%9HOME/.m2/settings.xml 文 件 ， 加 上 下 面 这 些 配置 ， 
为 各 种 元 素 填 上 恰当 的 什 。 

<PDroxies> 

<DIOXYy> 
cactivestruec/actiVves 
protocol></protocol> 
<USername></Username> 
<pageword></pageword> 
<host ></host> 
Ports</port> 

</proxy> 

</proxies> 


重新 运行 这 个 目标 ，BUILD SUccEss 消 息 如 期 而 至 。 


提示 跟 其 他 Maven 构 建 周 期 目标 不 同 ,clean 不 会 自动 调用 ,如 果 你 想 清 除 上 一 次 构建 产生 的 
工件 ， 必 须 把 clean 目 标 包 括 在 内 。 


现在 已 经 把 以 前 构建 的 残留 物 部 清除 掉 了 ， 一 般 接 下 来 要 执行 的 构建 周期 目标 是 编 刘 
代码 。 
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A.3.3 compile 


compile 目 标 用 pom.xml 文 件 中 的 compiler 插 件 配置 编译 在 src/main/java 、src/main/scala 和 
src/main/groovy 目 录 下 的 源码 。 这 实际 上 是 将 compile-scoped 的 依赖 项 加 到 cLAassPATH 上 执行 
Java、Scala 和 Groovy 编 译 器 ( javac、scalac 和 groovyc )。Maven 也 会 处 理 src/main/resources 
下 的 资源 ， 确 保 它们 出 现在 编译 时 的 cLAssPATH 中 。 

编 详 好 的 类 会 放 到 targetclasses 目 录 下 。 要 看 实际 效果 ， 请 执行 下 面 的 目标 : 

mn compile 


compile 目 标 执行 起 来 应 该 很 快 ， 控 制 台 的 输出 看 起 来 应 该 像 下 面 这 样 。 


[INFO] [compiler:compile 1execution: default-compile}] 

[INFO] Compiling 119 source files to 
C:\Projects\workspace3.6\code\trunk\target\classes 

[INFO] [scala:compile |execution: default}] 

[INFO] Checking for multiple versions of scala 

[LINFO] includes = [**/* ,acala,**/*.Java,] 

[INFO] excludes = [|] 

[INFO] C:\Projecta\workspace3.6\code\trunk\srcec\main\java:-1: infoe: compiling 

[INFO] C:\Projectae\workaspace3.6\code\trunk\target \generated-sources\groovy- 
stubs\main:-1: info: compiling 

[INFO] C:\Projects\workspace3.6\code\trunk\src\main\groovy:-1: info: 
compiling 

[INFO] C:\Projects\workspace3.6\code\trunk\src\main\scala:-1: info: compiling 

[INFO] Compiling 143 source files to 
C:\Projects\workspace3.6\code\trunk\target \classes at 1312716331031 

[INFO] prepare-complile in 0 s 

[INFO] compile in 1i2 8 

[INFO] [groovy:compile lexecution: default}] 

[INFO] Compiled 26 OO classes 


[INFO] Total time: 43 seconds 
[INFO] Finished at: Sun Aug 07 12:25:44 BST 2011 
[INFO] Final Ot 33M,79M 


在 这 一 阶段 ， 在 src/test/java、src/test/scala 和 |src/test/groovy 日 录 下 的 测试 类 还 没有 编译 。 尽 管 
针对 它们 有 专门 的 test-compile 目 标 , 但 更 典型 的 方式 是 让 Maven 运 行 test 目 标 。 


A.3.4 test 


运行 cest 目 标 能 看 到 Maven 的 构建 周期 的 真实 效果 。 在 要 求 Maven 测 试 时 ， 它 知道 自己 需要 
把 之 前 的 构建 周期 目标 全 都 执行 过 之 后 才能 成 功 运行 test 目标 ( 包括 compile、test-compile， 
还 有 很 多 其 他 的 )。 

Maven 会 通过 Surefire 插 件 ， 用 pom.xml 中 配置 为 test-scoped 依 束 项 的 测试 提供 者 (此 仆 为 
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JUnit ) 运行 测 坛 。Maven 不 仅 运 行 测 试 ， 还 会 生成 报告 文件 ， 供 以 后 进行 分 析 ， 调 研 失 败 测试 并 
收集 测试 指标 。 

要 看 实际 效果 ， 执 行 如 下 目标 : 

mn clean 七 已 号 七 

Maven 一 旦 完成 测试 失 的 编译 和 运行 ， 承 应 该 能 看 到 类 似 下 面 这 种 输出 的 报告 。 


com.Java7developer.chapterl1ll1l.listing 11 3.TicketRevenueTest 

Tests run: 5, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 0 sec 
Running com.javaTdeveloper.chapteril,.listing 11 4.TicketRevenueTegst 

Tests run: 5S5, Failuyures: 0, Errors: 0, Skipped: 0, Time elapsed: 0 sec 
Running com.Java7Tdeveloper.chapterll.listing 11 5S5.TicketTest 

Teasta run: 1, Failures: 0, Errora: 0, Skipped: 0, Time elapsed: 0.015 gec 


Regults : 


Tests run: 20, Fallures: 0, Errors: 0, Skipped: 

[INFO] ============ = = 一 = 一 = 一 一 = 一 一 一 一 
[LINEO BUILD SUCCESSFUL 

[INEFO]-=-=--=------------------------=--- 二 二 二 一 一 = 一 = 一 一 一 一 一 一 一 
[INFO] Total time: 16 seconds 

[INFO] Finished az Wed Jul 06 13:50:07 BST 2011 

[INFO] Final jb 24M/S8M 

[INFO] ======-=-- 


测试 结 来 保存 在 target/surefire i 你 现在 可 以 去 看 看 这 个 文本 文件 ， 能 看 到 测试 成 功 
通过 了 。 


A.4 小 结 


如 果 能 一 边 看 书 一 边 运 行书 中 的 源码 示例 , 你 会 对 书 中 的 内 容 有 更 深刻 的 认识 。 如 果 你 喜欢 
冒险 ， 还 可 以 改 一 改 我 们 的 代码 ， 甚 至 加 一 些 新 代码 ， 然 后 以 相同 的 方式 编译 和 测试 。 

像 Maven 3 这 样 的 构建 工具 ， 其 底层 实现 复杂 得 超 乎 想象 。 如 果 你 想 深入 了 解 这 一 主题 ， 请 
阅读 第 12 竟 ， 它 讨论 了 构建 和 持续 集成 方面 的 内 容 。 
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Java7 NIO.2 类 库 在 循环 遍历 目录 和 其 他 类 似 任 务 中 用 glob 模 式 执 行 过 滤 操 作 ， 参 见 第 2 章 。 


B.1 glob 模式 语法 


模式 比 正 则 表达 式 简 单 ， 其 基本 规则 如 表 B-1 所 示 。 
表 B-1 glob 模 式 语法 


语 法 描 述 
匹配 0 或 更 多 个 字符 
Es 跨越 目录 匹配 0 或 更 多 个 字符 
? 完全 匹配 单个 字符 
(] 限定 一 个 子 模式 集合 ， 0 比如 匹配 模式 A、B 或 C 等 
[] 匹配 一 组 字符 中 的 单个 字符 ， 或 者 如 果 字 符 间 有 连 字符 ( - ) ， 则 匹配 其 所 限定 范围 的 字符 
\ 转 义 符 ， 在 匹配 *、? 或 \ 之 类 的 特殊 字符 时 使 用 


要 进一步 了 解 glob 模 式 语法 ,请 参见 Oracle 的 在 线 Java 教 程 ( http://docs.oracle.com/javase/tutorial/ 
essential/io/fileOps.html#glob ) 及 Filesystem 类 的 Java 文 档 。 


B.2 glob 模式 示例 
一 些 使 用 glob 模 式 的 基本 例子 有 时 被 称 为 globbing， 如 表 B-2 所 示 。 
表 B-2 glob 模式 示 例 


语 法 描述 
:java 匹配 所 有 以 java 结 尼 的 字符 串 ， 比 如 Listing 2 1.java 
?3 匹配 任意 两 个 字符 ， 比 如 ab 或 xl 
[0-9] 匹配 0 到 9 之 间 的 任意 数字 
{groovy, scala}.* 匹配 所 有 以 groovy. 或 scala. 开 头 的 字符 串 ， 比 如 scala.txit 或 groovy.pdf 
[Ja-z, A-Z 匹配 一 个 大 与 或 小 与 的 美文 字符 
\\ 匹配 \ 字 符 
/usr/home/** 匹配 所 有 以 iusrihome/ 开 类 的 字符 帅 ， 比 如 /usr/home/karianna 或 /usr/home/karianna/docs 
只 自在 读书 @3 
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要 查看 更 多 glob 模 式 匹 配 的 例子 ， 请 参见 Oracle 的 在 线 Java 教 程 及 Filesvstem 类 的 Java 
文档 。 


敬告 “Java 7 规范 定义 了 自己 的 glob 语 义 ( 而 不 是 采用 已 有 的 标准 )。 有 些 可 能 会 变 成 给 程序 员 
控 的 坑 ， 特 别 是 在 Unix 上 。 比 如 说 ， 同 样 是 rm *， 在 Java 7 中 会 移 除 以 点 〔.) 开头 的 文 
人 忻 ， 而 在 Unix 的 rm/glob 中 则 不 会 移 除 这 样 的 文件 。 


山 在 Unix 的 glob 模 式 中 ， 如 果 文 件 名 以 “.” 开 头 ， 则 这 个 字符 必须 显 式 匹 配 。 因 此 rm * 不 会 称 除 .profile， 并且 
tar c * 也 不 会 归档 所 有 文件 ， 用 tar ec. 会 更 好 。 一 一 译 者 注 
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本 附录 涵盖 了 三 种 JVM 诸 言 ( Groovy、Scala 和 Clojure ) 以 及 Groovy 的 Web 框 架 ( Grails ) 的 
下 载 及 安装 指导 ， 它 们 各 自分 别 在 第 8 章 、 第 9 章 、 第 10 章 和 第 13 章 讨论 。 
C.1 Groovy 


装 Groovy 相 当 简 单 , 但 如 果 你 对 设置 环境 变量 不 部 ,或 3 
得 这 个 指南 很 有 帮助 。 


刚 接触 某 一 操作 系统 ， 你 应 该 会 觉 


C.1.1 下 载 Groovy 


请 先 访问 http://groovy.codehaus.org/Download 下 载 最 新 的 稳定 版 Groovy。 我 们 的 例子 用 的 是 
Groovy 1.8.6， 所 以 推荐 你 下 载 groovy-binary-1.8.6.zip 文 件 。 然 后 把 下 载 好 的 压缩 文件 解压 到 选 定 
的 目录 中 。 


警告 跟 很 多 Java/JVM 相 美 软 忻 的 安装 一 样 ， 在 安装 Groovy 的 目录 名 称 中 也 不 要 有 空格 ， 否 则 
可 能 会 出 现 PATH 和 CLASSPATH 和 错误。 比如 说 ， 如 果 你 用 的 是 Windows 操 作 系 统 ， 不 要 把 
Groovy 装 在 C:\Program Files\Groovy\ 这 样 的 目录 中 。 


剩 下 没 几 步 了 ， 接 下 来 需要 设置 环境 变量 。 


C.1.2 ” 安 浪 Groovy 


完成 下 载 和 解压 后 , 需要 设置 三 个 环境 变量 以 有 效 运 行 Groovy。 我 们 会 看 看 基于 POSIX 的 操 
作 系 统 ( Linux、Unix 和 Mac OS 义 ) 以 及 微软 Windows。 

1, 基于 POSIX 的 操作 系统 ‘(Linux、Unix、Mac OS X) 

在 一 个 基于 POSIX 的 操作 系统 上 ， 在 哪里 设置 操作 系统 通常 取决 于 打开 终端 窗口 时 运行 的 
shell。 表 C-1 中 包含 了 各 种 POSIX 操 作 系 统 shell 中 常见 的 用 户 shell 配 置 文件 的 名 称 及 位 置 。 
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表 C-1 用 户 shell 配 置 文件 的 常见 位 置 
Shell 文件 位 置 
bash ~/ bashrc 和 /或 ~/.profile 
Kom (ksh) ~/.kshrc 利 /或 ~/ .profile 
古 -/ paofile 
Mac OS X ~/.bashrec 和 /或 ~./ .profile 和 /或 ~./bash profile 


用 你 喜欢 的 编辑 器 打开 用 户 shell 配 置 文件 , 加 上 三 个 环境 变量 ; GROOVY_HOME、 JAVA_HOME 
和 PRaATH。 

需要 先 设 置 环境 变量 GROOVY_HOME。 加 上 下 面 这 一 行 ， 用 Groovy 文 件 的 真实 位 置 ( 即 解压 
文件 的 位 置 ) 换 掉 < 安 沪 目 录 >: 

GROOVY HOME=<installation directory> 

在 下 面 的 例子 中 ， 我 们 将 Groovy 解 压 到 本 /opt/groovy-1.8.6 中 : 

GROOVY HOME=/opt/groovy-1.8.6 

Groovy 需 要 Java JDK 才 能 运行 ,任何 大 于 1.5 的 版 本 都 行 ( 此 时 你 很 可 能 已 经 装 上 JDK 1.7 了 )。 
你 还 需要 确保 环境 变量 JAVA_HOME 已 经 设置 好 了 。 如 果 你 已 经 装 好 了 Java， 这 个 可 能 也 已 经 设置 
好 了 ， 如 果 还 没有 ， 可 以 添上 下 面 这 行 : 

JAVA HOME=<path to where Java is inatalled> 

在 下 面 的 例子 中 ,我们 将 JAVA_HOME 设 置 为 /opt/java/java-1.7.0: 

JAVA HOME=/opt /java/java-1.7.0 

最 后 ， 要 能 在 命令 行 中 的 任何 位 置 执行 Groovy 相 关 命令 ， 所 以 得 把 GROOVY_HOME /bin 加 到 
PATH 中 : 


PATH=$PATH: $GROOVY HOME/bin 
保存 用 户 shell 配 置 文件 ， 在 下 次 启动 新 shell 时 ， 这 三 个 变量 就 会 生效 。 现 在 为 了 确保 基本 安 
汶 可 以 正常 工作 ， 可 以 在 命令 行 中 执行 带 -version 参 数 的 groovy 命 今 : 


rooVY -Vers ion 
Groovy Version: 1.8.6 JVM: 1.7.0 


在 基于 POSIX 操 作 系 统 上 安装 Groovy 就 完成 了 。 现 在 你 可 以 回 到 第 8 章 ， 编 译 并 运行 Groovy 
代码 去 了 1 

2. Windows 

在 Windows 中 , 设置 环境 变量 最 好 的 方式 是 通过 管理 计算 机 的 GUI。 请 按照 下 面 这 些 步骤 操作 ， 

(1) 右键 点 击 “ 我 的 电脑 "， 然 后 点 击 “ 属 性 ”; 

(2) 选择 “高 级 ”选项 卡 ; 

G3) 点击“ 环境 变量 ”; 

(4) 点 击 “ 新 增 ” 添 加 新 的 变量 名 称 和 值 。 
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现在 需要 设置 环境 变量 sGRoovY_HoME。 加 上 下 面 这 一 行 ， 用 Groovy 文 件 的 真实 位 置 ( 即 解 
压 文件 的 位 置 ) 换 掉 < 安装 目录 >: 

GROOVY HOME=< installation dire ctoOrYy> 

在 下 面 的 例子 中 ， 我 们 将 Groovy 解 压 到 了 C:\languages\groovy-1.8.6 中 : 

GROOVY HOME=C:\languages\groovy-1.8.6 

Groovy 需 要 Java JDK 才 能 运行 ,任何 大 于 1.5 的 版 本 都 行 ( 此 时 你 很 可 能 已 经 装 上 JDK 1.7 了 了 )。 
你 还 需要 确保 环境 变量 JAVA_HOME 已 经 设置 好 了 。 如 果 你 已 经 装 好 了 Java, 这 个 可 能 也 已 经 设置 
好 了 ， 如 果 还 没有 ， 可 以 添上 下 面 这 行 : 

JAVA HOME=<path to where Java is installed> 

在 下 面 的 例子 中 ， 我 们 将 JAVA_HOME 设 置 为 C:\Java\jdk-1.7.0: 

JAVA HOME=C:\Java\jdk-1.7.0 

要 能 在 命令 行 中 的 任何 位 置 执 行 Groovy 相 关 命 令 ， 所 以 得 把 GROOVY_HOME /bin 加 到 
PATH 中 ， 

PATH=%PATHS ; $GROOVY HOMESNVbin 

一 直 点 击 “ 确 定 ” 直 到 退出 “我 的 电脑 ”的 管理 界面 。 在 下 次 启动 新 命令 行 时 ， 这 三 个 变量 
就 会 生效 ,现在 为 了 确保 基本 安装 可 以 正常 工作 , 可 以 在 命令 行 中 执行 带 -version 参 数 的 groovy 
命令 : 


roovy =Version 
Groovy Verslon: 1.8.6 JVM: 1.7.0 


在 Windows 上 安装 Groovy 就 完成 了 。 现 在 你 可 以 回 到 第 8 章 ， 编 译 并 运行 Groovy 代 码 去 了 1! 


C.2 Scala 


Scala 环 境 可 从 www.scala-lang.org/downloads 下 载 。 写 本 书 时 的 版 本 是 2.9.1, 但 在 你 读 到 这 
儿 时 可 能 已 经 有 新 版 本 发 布 。Scala 确 实 倾向 于 在 发 布 新 版 本 时 引入 语言 的 新 变化 ， 所 以 如 果 
你 发 现 某 些 示例 在 ( 较 新 的 ) Scala 上 不 能 用 ， 请 认真 检查 语言 的 版 本 ， 并 确保 你 为 本 书 装 的 
十 2.9.]1。 
Windows 用 户 应 该 下 载 .zip 版 本 的 ， 而 基于 Unix 系 统 ( 包括 Mac 和 Linux ) 的 用 户 应 该 下 载 ,tgz 
版 本 的 。 解 压 文件 ， 放 到 指定 的 位 置 。 跟 Groovy 一 样 ， 应 该 避免 名 称 中 包含 宝 格 的 目录 。 

有 几 种 办 法 可 以 在 你 的 机 大 上 设置 Scala。 最 简单 的 可 能 就 是 设 一 个 SCALA_HOME 环 境 变 量 指 
回 你 安装 S$cala 的 目录 。 然 后 根据 你 的 操作 系统 ， 按 照 C.1.2 节 的 指令 ( 安装 Groovy 的 )， 把 
GROOVY_HOME 全 换 成 SCRALA_HOME。 

下 完成 环境 的 配置 后 ， 可 以 在 命令 行 中 键 人 Ascala， 应 该 可 以 打开 Secala 交 互 式 会 话 。 如 果 没 
开 ， 说 明 环 境 配 置 得 不 正确 ， 应 该 重新 试 一 次 ， 确 保 ScALA_HOME 和 PaTH 的 设置 是 正确 的 。 

现在 应 该 可 以 运行 第 9 童 的 Scala 代 码 清 单 和 交互 式 的 代码 片段 了 。 
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C.3 Clojure 


要 下 载 Clojure， 请 访问 http://clojure.org/ 找 到 包含 最 新 稳定 版 的 zip 文 件 。 我 们 在 示例 中 用 的 
是 Clojure 1.2， 所 以 如 果 你 用 的 版 本 比较 新 ， 请 注意 它们 可 能 会 和 销 有 差异 。 

解压 刚 下 载 的 文件 ， 并 进入 它 刚 创建 好 的 目录 。 假 定 JAVA_HOME 已 经 设置 好 了 ，java 也 在 
PATH 上 ， 那么 现在 你 应 该 可 以 像 第 10 章 那样 运行 简单 的 REPL 了 ， 像 这 样 ; 

java -cp clojure.jar clojure.main 

Clojure 跟 这 个 附录 里 的 前 两 种 新 语言 不 太一 样 , 要 用 这 门 语言 真 的 内 需要 clojure.jar 文 件 。 不 
用 像 Groovy 和 Scala 那 样 设 置 任 何 环境 变量 。 

在 学 习 Clojure 时 ， 最 简单 的 可 能 就 是 用 REPL。 当 你 开始 考虑 用 Clojure 做 生产 环境 的 部 署 时 ， 
需要 用 一 个 恰当 的 构建 工具 ( 比如 第 12 章 介绍 的 Leiningen ) 来 管理 应 用 的 部 署 ， 以 及 Clojure 本 身 
的 安装 ( 从 远程 Maven 资 源 库 下 载 这 个 JAR 文 件 )。 

Clojure 的 基本 安装 有 些 局 限 性 ， 但 好 在 有 几 个 非常 好 的 Clojure 跟 IDE 的 集成 。 如 果 你 用 的 是 
Eclipse， 我 们 均 心 向 你 推荐 Eclipse 的 Counterclockwise 搬 件 ， 它 很 实用 ， 也 非常 容易 设置 。 

开发 经 验 稍微 丰富 一 点 非常 有 用 ， 因 为 在 简单 的 REPL 中 开发 大 量 代码 可 能 有 点 容易 分 散人 
的 注意 力 。 但 对 于 很 多 应 用 程序 ( 特别 是 你 正在 学 的 ) 来 说 ， 基 本 的 REPL 就 足够 了 。 


C.4 Grails 

装 Grails 相 当 简 单 ， 但 如 果 你 对 设置 环境 变量 不 熟 ， 或 者 刚 接触 某 一 操作 系统 ， 应 该 会 觉得 
这 个 指南 很 有 帮助 。www.grails,org/installation 上 有 完整 的 安装 指导 。 
C.4.1 下 载 Grails 

请 先 访问 www.grails.org 下 载 最 新 稳定 版 Grails。 我 们 在 本 书 中 用 的 版 本 是 2.0.1。 下 载 好 后 ， 
请 把 压缩 文件 解压 到 选 定 的 目录 中 。 


警告 ” 跟 很 多 Java/JVM 相 关 软 件 的 安装 一 样 ， 在 安装 Grails 的 目 录 名 称 中 不 要 有 空格 ， 否则 可 能 
会 出 现 PATH 和 CLaSSPRTH 错 误 。 比 如 说 ， 如 果 你 用 的 是 Windows 操 作 系 统 , 不 要 把 Grails 
装 在 C:\Program Files\Grails \ 这 样 的 目录 中 。 
接 下 来 需要 设置 环境 变量 。 
C.4.2 安装 Grails 


在 完成 下 载 和 解压 后 ， 需 要 设置 三 个 环境 变量 以 有 效 运 行 Grails。 我 们 会 看 看 在 基于 POSIX 
的 操作 系统 (Linux、Unix 和 Mac OSX ) 以 及 微软 Windows 中 如 何 设置 环境 变量 。 
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1. 基于 POSIX 的 操作 系统 (Linux、Unix、Mac OS X) 
在 一 个 基于 POSIX 的 操作 系统 上 ， 在 哪里 设置 操作 系统 通常 取决 于 打开 终 疹 窗口 时 运行 的 
shell。 表 C-2 中 包含 了 各 种 POSIX 操 作 系 统 shell 中 常见 的 用 户 shell 配 置 文件 的 和 名 称 及 位 置 。 
表 C-2 用 户 shell 配 置 文件 的 常见 位 置 
Shell 文件 位 置 
bash -/bashre 和 /或 -Apmfile 


Korn (ksh) -人 /kshre 和 /或 -/.profile 
sh ~/-profile 
Mac OS XA ~/.bashre 科 |/ 或 =f .profile 和 /或 ~./bash_profile 


用 你 喜欢 的 编辑 器 打开 用 户 shell 配 置 文件 , 加 上 三 个 环境 变量 : GRAILS_HOME、 JAVA_HOME 
和 PATH。 

需要 先 设置 环境 变量 GRAILS_HOME。 加 上 下 面 这 一 行 ， 用 Grails 文 件 的 真实 位 置 ( 即 解 压 文 
件 的 位 置 ) 换 掉 < 安装 目录 > 

GRAILS HOME=<installation directory> 

在 下 面 的 例子 中 ， 我 们 将 Grails 解 压 到 了 /opt/grails-2.0.1 中 : 

GRAILS HOME=/opt/grails-2.0.1 

Grails 需 要 JavaJDK 才 能 运行 。 任 何 大 于 1.5 的 版 本 都 行 ( 此 时 你 很 可 能 已 经 波 上 JDK 1.7 了 了 )。 
还 需要 确保 环境 变量 JaVa_HOME 已 经 设置 好 了 。 如 果 已 经 装 好 了 Java, 这 个 可 能 也 已 经 设置 好 了 ， 
如 果 还 没有 ， 可 以 添上 下 面 这 行 : 

JAVA HOME=<path to where Java is installed> 

在 下 面 的 例子 中 ， 我 们 将 JAVA_HOME 设 置 为 /opt/java/java-1.7.0: 

JAVA HOME=/opt/jJava/java-1.7.0 

最 后 ， 要 能 在 命令 行 中 的 任何 位 置 执行 Grails 相 关 命令 ， 所 以 得 把 cGRATLs_HoME/bin 加 到 
PATH 中 : 

PATH=$PATH: $GRATLS HOME /bin 

保存 用 户 shell 配 置 文件 ， 在 下 次 启动 新 shell 时 ， 这 三 个 变量 就 会 生效 。 现 在 为 了 而 
装 可 以 正常 工作 ， 可 以 在 命令 行 中 执行 带 -version 和 参数 的 grails 合 令 : 


qrails =-vergsion 


人保 基本 安 


Gralls version: 2.0.1 

在 基于 POSIX 操 作 系 统 上 安装 Grails 就 完成 了 。 现在 可 以 回 到 第 13 章 开始 你 的 第 一 个 Grails 项 
目 了 ! 

2. Windows 

在 Windows 中 ， 设 置 环境 变量 最 好 的 方式 是 通过 管理 计算 机 的 GUI。 请 按照 下 面 这 些 步 又 
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探 作 : 

(1) 右键 点 击 “ 我 的 电脑 "， 然 后 点 击 “属性 ”; 

(2) 选择 “高 级 ”选项 卡 ; 

G3) 点击“ 环境 变量 ”; 

(4) 点 击 “ 新 增 ” 添 加 新 的 变量 名 称 和 值 。 

现在 需要 设置 环境 变量 GRAILS_HOME。 加 上 下 面 这 一 行 ， 用 Grails 文 件 的 真实 位 置 ( 即 解压 
文件 的 位 置 ) 换 掉 < 安 装 目 录 > 

GRAILS HOME=<installation directory> 

在 下 面 的 例子 中 ， 我 们 将 Grails 解 压 到 了 C:\languages\grails-2.0.1 中 : 

GRAILS HOME=C:\languages\grails-2.0.1 

Grails 需 要 JavaJDK 才 能 运行 。 任 何 大 于 1.5 的 版 本 都 行 ( 此 时 你 很 可 能 已 经 装 上 JDK 1.7 了 )。 
还 需要 确保 环境 变量 JAVA_HOME 已 经 设置 好 了 。 如果 已 经 装 好 了 Java, 这 个 可 能 也 已 经 设置 好 了 ， 
如 果 还 没有 ， 可 以 洪 上 下 面 这 行 : 

JAVA HOME=<path to Where Java 1s installed> 

在 下 面 的 例子 中 ， 我 们 将 JAVA_HOME 设 置 为 C:\Java\jdk-1.7.0: 

JAVA HOME=C:\Java\jdk-1.7.0 

要 能 在 命令 行 中 的 任何 位 置 执行 Grails 相 关 命 令 ， 所 以 得 把 GRAILS_HOME/bin 加 到 PATH 中 : 

PATH=$%PATHSY ; GRAILS HOMES'‘\bin 

一 直 点 击 “ 确 定 ” 直 到 退出 “我 的 电脑 ”的 管理 界面 。 在 下 次 启动 新 命令 行 时 ， 这 三 个 变量 
束 会 生效 。 现 在 为 了 确保 基本 安装 可 以 正常 工作 ， 可 以 在 命令 行 中 执行 带 -version 参 数 的 
grails 命 邻 : 

grails -version 


Gralils version: 2.0.1 


在 Windows 上 安装 Grails 就 完成 了 。 现 在 可 以 回 到 第 13 童 开始 你 的 第 一 个 Grails 项 目 了 | 
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本 附录 讲解 Jenkins 的 下 载 和 安装 ， 第 12 章 要 用 到 它 。Jenkins 的 下 载 和 安装 很 简单 。 如 果 你 确 
实 磁 到 了 困难 ， 它 的 维基 页 面 https://wiki.jenkins-ci.org/display/JENKINS/Meet+Jenkins 应 该 能 提供 
帮助 。 


D.1 下 载 Jenkins 


可 以 从 http://mirrors.jenkins-ci.org/ 上 下 载 Jenkins。 第 12 章 的 例子 中 用 的 是 Jenkins 1.424。 
常见 的 跟 操 作 系 统 无 关 的 Jenkins 安 装 方式 是 用 jenkins.war 包 。 但 如 果 你 不 太 清楚 如 何 运行 目 
己 的 Web 服 务 器 ( 如 Apache Tomcat 或 Jetty )， 可 以 根据 自己 的 操作 系统 下 载 独立 的 安装 包 。 


提示 Jenkins 团 队 推出 新 版 本 的 频率 令 人 印象 深刻 ， 对 于 那些 想 有 更 稳定 的 CI 服务 器 的 团队 来 
说 ， 这 可 能 让 他 们 觉得 内 心志 下 。Jenkins 团 队 注 意 到 了 这 小 问题 ， 所 以 他 们 现在 推出 了 
一 个 长 期 支持 版 本 【Long Term Support，LTS )- 


接 下 来 ， 你 需要 按 几 个 简单 的 步 又 来 安装 Jenkins。 
D.2 安装 Jenkins 


不 管 选 的 是 WAR 文 件 还 是 独立 安装 包 , 下 载 完 之 后 需要 把 Jenkins 装 上 。Jenkins 要 有 JavaJDK 
才能 运行 ， 任 何 大 于 1.5 的 版 本 都 行 ( 此 时 你 很 可 能 已 经 装 上 JDK 1.7 了 )。 我 们 会 先 介 绍 WAR 的 
安装 ， 然 后 是 独立 安装 包 。 

D.2.1 运行 WAR 文件 

可 以 在 命令 行 上 运行 下 面 的 命令 直接 执行 Jenkins WAR 文 件 ， 这 样 安装 Jenkins 非 常 快 : 

java -jar jenkines.war 

这 种 安装 方法 仅 适 用 于 Jenkins 的 快速 试用 , 因为 它 很 难 对 其 他 相关 的 Web 服 务 器 参数 进行 配 
置 ， 所 以 运行 起 来 可 能 不 会 那么 顺畅 。 
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D.2.2 安装 WAR 文件 


对 于 更 长 久 的 安装 ， 需 要 把 WAR 文 件 部 署 到 你 使 用 的 Web 服 务 器 上 ( 支持 Java Web 应 用 )。 
对 于 java7developer 项 目 而 言 ， 我 们 只 要 把 jenkins.war 文 件 黏 贴 到 Apache Tomcat 7.0.16 服 务 船 的 
webapps 目 录 下 就 行 了 。 

如 果 你 对 WAR 文 件 和 能 支持 Java Web 应 用 的 Web 服 务 器 不 熟悉 ， 可 以 用 独立 安装 包 。 


D.2.3 ” 安 闭 独立 安 效 包 


独立 安装 包装 起 来 也 很 简单 。 要 在 Windows 上 安装 ， 先 解压 jenkins-<version>.zip 文 件 ， 然 后 
运行 exe 或 msi 安 装 文件 。 要 在 各 种 Linux 上 安装 , 需要 运行 合适 的 yum 或 rpm 包 管理 器 。 而 对 于 Mac 
OS X 而 言 ， 要 运行 pkg 文 件 。 

无 论 哪 种 情况 ,都 可 以 选择 安装 文件 给 出 的 默认 选项 , 或 者 按 自己 的 想法 选择 安装 中 
他 设置 。 

默认 情况 下 ，Jenkins 会 把 配置 信息 和 任务 保存 在 用 户 的 主 目录 (我们 用 $USER 表 示 ) 中 
的 .jenkins 目 录 下 。 要 编辑 UI 之 外 的 任何 配置 ， 也 可 以 到 这 个 目录 下 。 


径 或 其 


D.24 Jenkins 的 首次 运行 


用 你 喜欢 的 浏览 器 访问 Jenkins 仪 表 板 ( 地 址 通常 是 http://localhost:8080/ 或 http://localhost: 
8080/jenkins )， 污 的 对 不 对 一 看 就 知道 了 。 安 装 正确 的 话 应 该 能 见 到 跟 图 D-1 类 似 的 界面 。 


donkan 


图 D-1 Jenkins 仅 表 板 
装 好 了 Jenkins， 现 在 可 以 开始 创建 Jenkins 任 务 了 。 请 回 到 第 12 章 关于 Jenkins 的 章节 去 吧 
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本 附录 涵盖 了 第 12 章 中 用 来 构建 java7developer 的 pom.xml 文 件 ， 在 pom.xml 文 件 的 重要 部 分 
上 上 展开， 以便 你 能 理解 整个 构建 。 基 本 项 目 信息 (12.1 节 ) 和 环境 配置 ( 代码 请 单 12-4 ) 在 第 12 
章 的 讨论 已 经 很 充分 了 ， 所 以 我 们 在 这 里 会 讨论 POM 的 下 面 两 个 部 分 : 

口 构建 配置 ; 

口 依赖 项 。 

我 们 先 从 最 长 的 那 一 部 分 开始 ， 即 构建 配置 。 


E.1 构建 配置 


构建 部 分 (<build> ) 包 会 一 些 插件 及 其 配置 信息 , 党 要 靠 它 们 来 执行 Maven 构 建 周期 目标 。 
对 于 大 多 数 项 目 来 说 ， 这 一 部 分 通 弟 都 相当 短 ， 因 为 一 般 用 默认 插件 的 默认 设置 就 够 了 。 但 对 于 
java7developer 项 目 而 言 ，<builda> 部 分 包含 了 几 个 覆盖 了 默认 设置 的 搬 件 。 我 们 之 所 以 这 样 做 ， 
是 为 了 让 java7developer 项 目 可 以 : 

D 构建 Java 7 代码 ; 

口 构建 Scala 和 Groovy 代 码 ; 

吕 运行 Java、Scala 和 Groovy 测 试 ; 

口 提供 Checkstyle 和 FindBugs 代 码 指 标 报告 。 

如 果 你 的 构建 中 还 有 更 多 需要 配置 的 地 方 ， 可 以 在 http://maven.apache.org/plugins/index.html 
找到 完整 的 插件 列表 。 

代码 清单 E-1 是 java 7 developer 项 目的 构建 配置 。 
代码 清单 E-1 POM: 构建 信息 


huilds 
<Dluginss 


<Dlugins 
<groupld>org.apache .maven.plugins</groupld> | oe 
<artifactIidsmaven-compiler=plugin</artifact1d> 和 指明 要 用 的 插件 


Version>2.3.2</Vvergions 
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<Configurations 
<S30Urce>1.7</S30Urce> 局 编译 Java 7 代码 
<target>l1.7</target> 
<BhowDeprecation>true</showDeprecation> | 
<ShowWarnings>strue< /showWarnings> 设置 编译 器 选项 
<forkstrue</fork> 
<executable>${(jdk.javac.fullpath}</executables 

</configurations 


ed 》 设 置 javac 的 路 径 
plugin> 
<groupIld>org.scala-tooles</groupId> 
<artifactIid>maven-scala-plugin</artifactId> 
<VerBion>2.14.1</version> 
<execut icons> 
executions 
Oalss 
<oOal>scompile</goal> 9 强制 Scala 编 译 
<Ioal>teastCompile</goal> 
</goals> 
/execution> 
</ Executionss 
<Configquration> 
<acalaVersion>2.9.0</scalaVersion> 
/confiqurations 
</plugins 


<Pplugin> 
<groupld>org.codehaus .gmaven< /groupId> 
<artifactIidsgmaven-plugin</artifactIids 
VerBionsl] .3</versions 
<dependencies> 
<dependency> 
<rouplId>org.codehaus .gmaven.runtime</groupId> 
<artifactIdsgmaven-runtime-1.7</artifactIds 
<VeErasionsl .de/Versions 
</dependency> 
</dependencies> 
二 全 XeCuLILOnS> 
<EXAECUtL1ON> 
<Configurations 
<providerSelection»>l1.7</providerSelection> 
</configuration> 
亏本 局 明生 > 
<gqoal>generateStubs<c/goal> 
<goalscompilez</goals 
<goal >generateTestStubs</goal> 
<aoal>testcCcompile</goal> 
</aqoals> 
</execution> 
/executionss 
</plugin> 


<Pplugins 
<groupId>org.codehauas .mojo< /groupIds 
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<artifactIidsproperties-maven-pluginc/artifactId> 
<Vergionsl1.0-aAlpha-2</Vergion> 
<EXeCULCions> 
<eXecCUtion> 
<phase>initialize</phase> 
<OOale> 
<oal>read-project-properties=</goal> 
</gqoalss 
<Configurations 
<filess 
<file>$ (basedir)} /build.properties</file> 
</filess 
</configurations 
/execution> 
</executions> 
</Pplugin> 
/pluginss 


plugins 
<groupld>org.apache .maven.plugins</groupIds 
<artifactId>maven-surefire-plugin</artifactIds 
<Version>2.9¢</version> 
<Configuration> 
<excCcludess> 
<eExCclude> 
com/java7rdeveloper/chapterll/listing 11 2 
ww /TicketRevenueTeest .java 
< /excludes 
<exClude> 
com/jJava7developer/chapterll/listing 11 7 
ss /TicketTest .iava 
/excludes 


排除 测试 


</excludeg> 
</configurations 
/plugins 


<PlUgin> 
<groupId>org.apache .maven.plugins</groupId> 
<artifactIidsmaven-checkstyle-plugin</artifactIds 
<Versgion>2.6</Vversions> 
<COnfigurations 
<includeTestSourceDirectory> 


> ] 在 测试 上 运行 Checkstyle 
</includeTestSourceDirectory> | 


</configqgurations 
</jplugins>s 
<plugin> 
<roupId>org.codehaus .moijo</groupId> 
<artifactId>findbugs-maven-plugin</artifactId> 
VeErgion>2.3.2</vergions 
<cConfigurations 


只 自 在 读书 @3 


WWwwW .Zizidiary .com 


附录 上 java7developer: Maven POM 393 


findbugaiXmlOutput>true</findbugaXmlOutput> 
<findbugsXmlWithMessages> | 生成 FindBugs 报 告 


true 
</findbugsXmlWithMessages> 
<XxXMmMlOutputstrue</xmlOutputs 
</configqguration> 
</plugin> 


</build>s 

因为 你 要 将 编译 Java 1.5 代 码 的 默认 行为 变 为 编译 Java 1.7@， 所 以 需要 指明 正在 使 用 ( 特定 
版 本 ) 的 Compiler ( 编译 器 ) 插件 @， 

因为 已 经 打破 了 惯例 ， 所 以 最 好 加 上 几 个 其 他 的 编译 器 警告 选项 轿 。 还 可 以 指明 Java 7 装 在 
哪里 介 。 要 想 让 Maven 得 到 javac 的 位 置 ， 只 要 将 与 操作 系统 对 应 的 sample_build.properties 男 存 
为 build.properties ， 并 修改 属性 jak .javac .fullpath 的 值 即 可 。 

为 了 使 用 Scala 捕 件 ， 需 要 确保 compile 和 testcompile 目 标 运行 时 Scala 搬 件 能 够 执行 @@?。 
用 Surefire 插 件 可 以 对 测试 进行 配置 。 在 这 个 项 目的 配置 中 ， 排 除了 几 个 故意 失败 的 测试 @ (你 
会 记 起 来 自 第 11 章 的 两 个 TDD 测 试 )。 

我 们 已 经 讨论 过 构建 部 分 了 ， 现 在 让 我 们 转 入 POM 中 的 男 一 个 关键 部 分 ， 依 赖 管理 。 


E.2 依赖 项 管理 


大 多 数 Java 项 目的 依 束 项 清单 都 很 长 ，java7developer 也 不 例外 。Maven 可 以 帮 你 管理 这 些 依 
下 项 ， 它 在 Maven Central Repository 中 存 了 数量 庞大 的 第 三 方 类 库 。 重 要 的 是 ， 这 些 第 三 - 方 类 库 
都 有 它们 目 己 的 pom.xml 文 件 ， 其 中 又 声明 了 它们 各 自 的 依赖 项 ，Maven 由 此 可 以 推断 出 你 还 需 
要 哪些 类 库 并 下 载 它 们 。 

最 初 会 用 到 两 个 主要 作用 域 是 compi1le 和 test”。 这 跟 把 JAR 文 件 放 到 cLAsspPATH 中 编译 代 
公然 后 运行 测试 效 来 是 完全 一 梓 的 。 

代码 清单 E-2 为 java7developer 项 目 pom 文 件 中 的 <dependencies> 部 分 。 


代码 清单 E-2 POM: 


<dependenciess 


dependencys 
<groupld>com.google.inject</groupId> | 
cartifactIld>gqguice</artifactId> I 工件 的 唯一 ID 
<VErgicon>3.0</vVergions 


< 号 COPe>ComplLe</ SCcope> 十 一 
</dapendency> 
dependencys 
. pO De 


<groupld>javax.inject</groupld> 
<artifactIlid>javax.inject</artifactIds 


山 希望 这 个 插件 的 后 续 版 本 能 自动 挂 到 这 些 日 标 上 。 
加 J2EE/JEE 项 目 中 通常 还 会 有 些 声 明 为 runtime 作 用 域 的 依赖 项 。 
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<artifactId>javax.inject</artifactId> 
Vergionsl</vergion> 
SCOpescompilee/scopes 

</dependaency> 

<dependency> 
<aroupId>org .codehaus .groovy</groupld> 
<artifactIid>sgroovy-all</artifactId> 
Versgions>l .0.6</versions 
Scope>compilec/scope> 

</dependency> 

<dependency> 
<JroupId>org.hibernate< /grouplIds 
AartifactIidshibernate-core<z/artifact1Ids> 
<Vversion>3.6.3.Final</version> 
<SCope>compile</scope> 

</dependency> 

-dependency> 
<roupIld>org .ow2 .asm< /gqroupId> 
<artifactlId>sasm</artifactId> 
Version>4.0</versions 
<BCOpe>compile</Scope> 

</dependency> 


dependency> 
<roupId>sijunit</grouplds> 
<artifactlId>ijunit</artifactId> 


<version>4.8.2</versions 9 test 作 用 域 
<Scope>test</Scope> 


</dependency> 
<dependency> 
<roupId>org .mockito</groupId> 
artifactIidsmockito-alle /artifactIids 
<Vversion>]1 .8.5</version> 
<SCODeE>test</ScODe> 
</dependency> 
<dependency> 
< 吕 roupId>org.scalatest</groupId> 
<artifactlid>scalatest 2.9.0</artifactId> 


<Version>1.6.1</Vversion> compile 作 用 域 
<SCOPe>compile</scope> ee ss 


</dependency> 

<dependency> 
<aroupld>org.hsqldbz /groupld> 
<artifactIdshsqldba<,/artifact1lds 
<VergBions2.2.4</Versaions 
<Scope>test</acope> 

</dependency> 

<dependency> 
<groupld>javassist</groupId> 
<artifactId>javassist</artifactId> 
Versions3.12.1.GAc /versions 
<SCope>test</Scope> 

</dependency> 

</dependencies> 
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为 了 让 Maven 找 到 我 们 引用 的 工件 ， 需 要 给 出 正确 的 <groupId> 、<artifactId> 和 
<version> 和 个。 我 们 在 前 面 说 过 了 , 将 <scope> 设 为 compile 因 会 将 那些 JAR 文 件 加 到 编译 代码 
所 用 的 CLassPaTH 中 。 

将 <scope> 设 为 test 因 会 确保 Maven 编 译 和 运行 测试 时 将 这 些 JAR 文 件 加 到 cLAsSsSPATH 
中 。 scalatest 类 库 是 其 中 比较 奇怪 的 , 它 应 该 放 在 test 作 用 域 中 , 但 要 放 在 compile 作 用 域 @ 才 
能 用 。” 

Maven pom.xml 文 件 并 不 像 我 们 所 期 望 的 那么 紧 资 ,但 我 们 执行 的 是 三 种 语言 的 构建 (Java、 
Groovy 和 Scala )， 还 能 生成 报告 。 和 希望 随 着 对 这 一 领域 的 工具 支持 不 断 改善 ，Maven 构 建 脚 本 能 
变 得 更 精简 。 


由 希望 这 个 插件 的 后 续 版 本 能 解决 这 个 问题 。 
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